Notifications: mark as read/unread and pin (#629)
* Use relative URLs * Notifications - Mark as read/unread * Feature of pinning a notification * On view issue, do not mark as read a pinned notification
This commit is contained in:
		
							parent
							
								
									cbf2a967c5
								
							
						
					
					
						commit
						769e0a3ea6
					
				| 
						 | 
				
			
			@ -589,7 +589,10 @@ func runWeb(ctx *cli.Context) error {
 | 
			
		|||
	})
 | 
			
		||||
	// ***** END: Repository *****
 | 
			
		||||
 | 
			
		||||
	m.Get("/notifications", reqSignIn, user.Notifications)
 | 
			
		||||
	m.Group("/notifications", func() {
 | 
			
		||||
		m.Get("", user.Notifications)
 | 
			
		||||
		m.Post("/status", user.NotificationStatusPost)
 | 
			
		||||
	}, reqSignIn)
 | 
			
		||||
 | 
			
		||||
	m.Group("/api", func() {
 | 
			
		||||
		apiv1.RegisterRoutes(m)
 | 
			
		||||
| 
						 | 
				
			
			
 | 
			
		|||
| 
						 | 
				
			
			@ -448,7 +448,7 @@ func (issue *Issue) ReadBy(userID int64) error {
 | 
			
		|||
		return err
 | 
			
		||||
	}
 | 
			
		||||
 | 
			
		||||
	if err := setNotificationStatusRead(x, userID, issue.ID); err != nil {
 | 
			
		||||
	if err := setNotificationStatusReadIfUnread(x, userID, issue.ID); err != nil {
 | 
			
		||||
		return err
 | 
			
		||||
	}
 | 
			
		||||
 | 
			
		||||
| 
						 | 
				
			
			
 | 
			
		|||
| 
						 | 
				
			
			@ -5,6 +5,7 @@
 | 
			
		|||
package models
 | 
			
		||||
 | 
			
		||||
import (
 | 
			
		||||
	"fmt"
 | 
			
		||||
	"time"
 | 
			
		||||
)
 | 
			
		||||
 | 
			
		||||
| 
						 | 
				
			
			@ -20,6 +21,8 @@ const (
 | 
			
		|||
	NotificationStatusUnread NotificationStatus = iota + 1
 | 
			
		||||
	// NotificationStatusRead represents a read notification
 | 
			
		||||
	NotificationStatusRead
 | 
			
		||||
	// NotificationStatusPinned represents a pinned notification
 | 
			
		||||
	NotificationStatusPinned
 | 
			
		||||
)
 | 
			
		||||
 | 
			
		||||
const (
 | 
			
		||||
| 
						 | 
				
			
			@ -182,13 +185,19 @@ func getIssueNotification(e Engine, userID, issueID int64) (*Notification, error
 | 
			
		|||
}
 | 
			
		||||
 | 
			
		||||
// NotificationsForUser returns notifications for a given user and status
 | 
			
		||||
func NotificationsForUser(user *User, status NotificationStatus, page, perPage int) ([]*Notification, error) {
 | 
			
		||||
	return notificationsForUser(x, user, status, page, perPage)
 | 
			
		||||
func NotificationsForUser(user *User, statuses []NotificationStatus, page, perPage int) ([]*Notification, error) {
 | 
			
		||||
	return notificationsForUser(x, user, statuses, page, perPage)
 | 
			
		||||
}
 | 
			
		||||
func notificationsForUser(e Engine, user *User, status NotificationStatus, page, perPage int) (notifications []*Notification, err error) {
 | 
			
		||||
func notificationsForUser(e Engine, user *User, statuses []NotificationStatus, page, perPage int) (notifications []*Notification, err error) {
 | 
			
		||||
	// FIXME: Xorm does not support aliases types (like NotificationStatus) on In() method
 | 
			
		||||
	s := make([]uint8, len(statuses))
 | 
			
		||||
	for i, status := range statuses {
 | 
			
		||||
		s[i] = uint8(status)
 | 
			
		||||
	}
 | 
			
		||||
 | 
			
		||||
	sess := e.
 | 
			
		||||
		Where("user_id = ?", user.ID).
 | 
			
		||||
		And("status = ?", status).
 | 
			
		||||
		In("status", s).
 | 
			
		||||
		OrderBy("updated_unix DESC")
 | 
			
		||||
 | 
			
		||||
	if page > 0 && perPage > 0 {
 | 
			
		||||
| 
						 | 
				
			
			@ -241,15 +250,53 @@ func getNotificationCount(e Engine, user *User, status NotificationStatus) (coun
 | 
			
		|||
	return
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
func setNotificationStatusRead(e Engine, userID, issueID int64) error {
 | 
			
		||||
func setNotificationStatusReadIfUnread(e Engine, userID, issueID int64) error {
 | 
			
		||||
	notification, err := getIssueNotification(e, userID, issueID)
 | 
			
		||||
	// ignore if not exists
 | 
			
		||||
	if err != nil {
 | 
			
		||||
		return nil
 | 
			
		||||
	}
 | 
			
		||||
 | 
			
		||||
	if notification.Status != NotificationStatusUnread {
 | 
			
		||||
		return nil
 | 
			
		||||
	}
 | 
			
		||||
 | 
			
		||||
	notification.Status = NotificationStatusRead
 | 
			
		||||
 | 
			
		||||
	_, err = e.Id(notification.ID).Update(notification)
 | 
			
		||||
	return err
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
// SetNotificationStatus change the notification status
 | 
			
		||||
func SetNotificationStatus(notificationID int64, user *User, status NotificationStatus) error {
 | 
			
		||||
	notification, err := getNotificationByID(notificationID)
 | 
			
		||||
	if err != nil {
 | 
			
		||||
		return err
 | 
			
		||||
	}
 | 
			
		||||
 | 
			
		||||
	if notification.UserID != user.ID {
 | 
			
		||||
		return fmt.Errorf("Can't change notification of another user: %d, %d", notification.UserID, user.ID)
 | 
			
		||||
	}
 | 
			
		||||
 | 
			
		||||
	notification.Status = status
 | 
			
		||||
 | 
			
		||||
	_, err = x.Id(notificationID).Update(notification)
 | 
			
		||||
	return err
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
func getNotificationByID(notificationID int64) (*Notification, error) {
 | 
			
		||||
	notification := new(Notification)
 | 
			
		||||
	ok, err := x.
 | 
			
		||||
		Where("id = ?", notificationID).
 | 
			
		||||
		Get(notification)
 | 
			
		||||
 | 
			
		||||
	if err != nil {
 | 
			
		||||
		return nil, err
 | 
			
		||||
	}
 | 
			
		||||
 | 
			
		||||
	if !ok {
 | 
			
		||||
		return nil, fmt.Errorf("Notification %d does not exists", notificationID)
 | 
			
		||||
	}
 | 
			
		||||
 | 
			
		||||
	return notification, nil
 | 
			
		||||
}
 | 
			
		||||
| 
						 | 
				
			
			
 | 
			
		|||
| 
						 | 
				
			
			@ -2712,6 +2712,12 @@ footer .ui.language .menu {
 | 
			
		|||
  float: left;
 | 
			
		||||
  margin-left: 7px;
 | 
			
		||||
}
 | 
			
		||||
.user.notification .buttons-panel button {
 | 
			
		||||
  padding: 3px;
 | 
			
		||||
}
 | 
			
		||||
.user.notification .buttons-panel form {
 | 
			
		||||
  display: inline-block;
 | 
			
		||||
}
 | 
			
		||||
.user.notification .octicon-issue-opened,
 | 
			
		||||
.user.notification .octicon-git-pull-request {
 | 
			
		||||
  color: #21ba45;
 | 
			
		||||
| 
						 | 
				
			
			@ -2722,6 +2728,9 @@ footer .ui.language .menu {
 | 
			
		|||
.user.notification .octicon-git-merge {
 | 
			
		||||
  color: #a333c8;
 | 
			
		||||
}
 | 
			
		||||
.user.notification .octicon-pin {
 | 
			
		||||
  color: #2185d0;
 | 
			
		||||
}
 | 
			
		||||
.dashboard {
 | 
			
		||||
  padding-top: 15px;
 | 
			
		||||
  padding-bottom: 80px;
 | 
			
		||||
| 
						 | 
				
			
			
 | 
			
		|||
| 
						 | 
				
			
			@ -85,6 +85,16 @@
 | 
			
		|||
            margin-left: 7px;
 | 
			
		||||
        }
 | 
			
		||||
 | 
			
		||||
        .buttons-panel {
 | 
			
		||||
            button {
 | 
			
		||||
                padding: 3px;
 | 
			
		||||
            }
 | 
			
		||||
 | 
			
		||||
            form {
 | 
			
		||||
                display: inline-block;
 | 
			
		||||
            }
 | 
			
		||||
        }
 | 
			
		||||
 | 
			
		||||
        .octicon-issue-opened, .octicon-git-pull-request {
 | 
			
		||||
            color: #21ba45;
 | 
			
		||||
        }
 | 
			
		||||
| 
						 | 
				
			
			@ -94,5 +104,8 @@
 | 
			
		|||
        .octicon-git-merge {
 | 
			
		||||
            color: #a333c8;
 | 
			
		||||
        }
 | 
			
		||||
        .octicon-pin {
 | 
			
		||||
            color: #2185d0;
 | 
			
		||||
        }
 | 
			
		||||
    }
 | 
			
		||||
}
 | 
			
		||||
| 
						 | 
				
			
			
 | 
			
		|||
| 
						 | 
				
			
			@ -1,7 +1,9 @@
 | 
			
		|||
package user
 | 
			
		||||
 | 
			
		||||
import (
 | 
			
		||||
	"errors"
 | 
			
		||||
	"fmt"
 | 
			
		||||
	"strconv"
 | 
			
		||||
	"strings"
 | 
			
		||||
 | 
			
		||||
	"github.com/Unknwon/paginater"
 | 
			
		||||
| 
						 | 
				
			
			@ -9,6 +11,7 @@ import (
 | 
			
		|||
	"code.gitea.io/gitea/models"
 | 
			
		||||
	"code.gitea.io/gitea/modules/base"
 | 
			
		||||
	"code.gitea.io/gitea/modules/context"
 | 
			
		||||
	"code.gitea.io/gitea/modules/setting"
 | 
			
		||||
)
 | 
			
		||||
 | 
			
		||||
const (
 | 
			
		||||
| 
						 | 
				
			
			@ -56,7 +59,8 @@ func Notifications(c *context.Context) {
 | 
			
		|||
		status = models.NotificationStatusUnread
 | 
			
		||||
	}
 | 
			
		||||
 | 
			
		||||
	notifications, err := models.NotificationsForUser(c.User, status, page, perPage)
 | 
			
		||||
	statuses := []models.NotificationStatus{status, models.NotificationStatusPinned}
 | 
			
		||||
	notifications, err := models.NotificationsForUser(c.User, statuses, page, perPage)
 | 
			
		||||
	if err != nil {
 | 
			
		||||
		c.Handle(500, "ErrNotificationsForUser", err)
 | 
			
		||||
		return
 | 
			
		||||
| 
						 | 
				
			
			@ -79,3 +83,32 @@ func Notifications(c *context.Context) {
 | 
			
		|||
	c.Data["Page"] = paginater.New(int(total), perPage, page, 5)
 | 
			
		||||
	c.HTML(200, tplNotification)
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
// NotificationStatusPost is a route for changing the status of a notification
 | 
			
		||||
func NotificationStatusPost(c *context.Context) {
 | 
			
		||||
	var (
 | 
			
		||||
		notificationID, _ = strconv.ParseInt(c.Req.PostFormValue("notification_id"), 10, 64)
 | 
			
		||||
		statusStr         = c.Req.PostFormValue("status")
 | 
			
		||||
		status            models.NotificationStatus
 | 
			
		||||
	)
 | 
			
		||||
 | 
			
		||||
	switch statusStr {
 | 
			
		||||
	case "read":
 | 
			
		||||
		status = models.NotificationStatusRead
 | 
			
		||||
	case "unread":
 | 
			
		||||
		status = models.NotificationStatusUnread
 | 
			
		||||
	case "pinned":
 | 
			
		||||
		status = models.NotificationStatusPinned
 | 
			
		||||
	default:
 | 
			
		||||
		c.Handle(500, "InvalidNotificationStatus", errors.New("Invalid notification status"))
 | 
			
		||||
		return
 | 
			
		||||
	}
 | 
			
		||||
 | 
			
		||||
	if err := models.SetNotificationStatus(notificationID, c.User, status); err != nil {
 | 
			
		||||
		c.Handle(500, "SetNotificationStatus", err)
 | 
			
		||||
		return
 | 
			
		||||
	}
 | 
			
		||||
 | 
			
		||||
	url := fmt.Sprintf("%s/notifications", setting.AppSubURL)
 | 
			
		||||
	c.Redirect(url, 303)
 | 
			
		||||
}
 | 
			
		||||
| 
						 | 
				
			
			
 | 
			
		|||
| 
						 | 
				
			
			@ -82,7 +82,7 @@
 | 
			
		|||
 | 
			
		||||
								{{if .IsSigned}}
 | 
			
		||||
									<div class="right menu">
 | 
			
		||||
										<a href="/notifications" class="ui head link jump item poping up" data-content='{{.i18n.Tr "notifications"}}' data-variation="tiny inverted">
 | 
			
		||||
										<a href="{{$.AppSubUrl}}/notifications" class="ui head link jump item poping up" data-content='{{.i18n.Tr "notifications"}}' data-variation="tiny inverted">
 | 
			
		||||
											<span class="text">
 | 
			
		||||
												<i class="octicon octicon-inbox"><span class="sr-only">{{.i18n.Tr "notifications"}}</span></i>
 | 
			
		||||
 | 
			
		||||
| 
						 | 
				
			
			
 | 
			
		|||
| 
						 | 
				
			
			@ -5,7 +5,7 @@
 | 
			
		|||
		<h1 class="ui header">{{.i18n.Tr "notification.notifications"}}</h1>
 | 
			
		||||
 | 
			
		||||
		<div class="ui top attached tabular menu">
 | 
			
		||||
			<a href="/notifications?q=unread">
 | 
			
		||||
			<a href="{{$.AppSubUrl}}/notifications?q=unread">
 | 
			
		||||
				<div class="{{if eq .Status 1}}active{{end}} item">
 | 
			
		||||
					{{.i18n.Tr "notification.unread"}}
 | 
			
		||||
					{{if eq .Status 1}}
 | 
			
		||||
| 
						 | 
				
			
			@ -13,7 +13,7 @@
 | 
			
		|||
					{{end}}
 | 
			
		||||
				</div>
 | 
			
		||||
			</a>
 | 
			
		||||
			<a href="/notifications?q=read">
 | 
			
		||||
			<a href="{{$.AppSubUrl}}/notifications?q=read">
 | 
			
		||||
				<div class="{{if eq .Status 2}}active{{end}} item">
 | 
			
		||||
					{{.i18n.Tr "notification.read"}}
 | 
			
		||||
					{{if eq .Status 2}}
 | 
			
		||||
| 
						 | 
				
			
			@ -30,34 +30,66 @@
 | 
			
		|||
					{{.i18n.Tr "notification.no_read"}}
 | 
			
		||||
				{{end}}
 | 
			
		||||
			{{else}}
 | 
			
		||||
				<div class="ui relaxed divided list">
 | 
			
		||||
				<div class="ui relaxed divided selection list">
 | 
			
		||||
					{{range $notification := .Notifications}}
 | 
			
		||||
						{{$issue := $notification.GetIssue}}
 | 
			
		||||
						{{$repo := $notification.GetRepo}}
 | 
			
		||||
						{{$repoOwner := $repo.MustOwner}}
 | 
			
		||||
 | 
			
		||||
						<div class="item">
 | 
			
		||||
							<a href="{{$.AppSubUrl}}/{{$repoOwner.Name}}/{{$repo.Name}}/issues/{{$issue.Index}}">
 | 
			
		||||
								{{if and $issue.IsPull}}
 | 
			
		||||
									{{if $issue.IsClosed}}
 | 
			
		||||
										<i class="octicon octicon-git-merge"></i>
 | 
			
		||||
									{{else}}
 | 
			
		||||
										<i class="octicon octicon-git-pull-request"></i>
 | 
			
		||||
									{{end}}
 | 
			
		||||
								{{else}}
 | 
			
		||||
									{{if $issue.IsClosed}}
 | 
			
		||||
										<i class="octicon octicon-issue-closed"></i>
 | 
			
		||||
									{{else}}
 | 
			
		||||
										<i class="octicon octicon-issue-opened"></i>
 | 
			
		||||
									{{end}}
 | 
			
		||||
						<a class="item" href="{{$.AppSubUrl}}/{{$repoOwner.Name}}/{{$repo.Name}}/issues/{{$issue.Index}}">
 | 
			
		||||
							<div class="buttons-panel right floated content">
 | 
			
		||||
								{{if ne $notification.Status 3}}
 | 
			
		||||
									<form action="{{$.AppSubUrl}}/notifications/status" method="POST">
 | 
			
		||||
										{{$.CsrfTokenHtml}}
 | 
			
		||||
										<input type="hidden" name="notification_id" value="{{$notification.ID}}" />
 | 
			
		||||
										<input type="hidden" name="status" value="pinned" />
 | 
			
		||||
										<button class="ui button" title="Pin notification">
 | 
			
		||||
											<i class="octicon octicon-pin"></i>
 | 
			
		||||
										</button>
 | 
			
		||||
									</form>
 | 
			
		||||
								{{end}}
 | 
			
		||||
								{{if or (eq $notification.Status 1) (eq $notification.Status 3)}}
 | 
			
		||||
									<form action="{{$.AppSubUrl}}/notifications/status" method="POST">
 | 
			
		||||
										{{$.CsrfTokenHtml}}
 | 
			
		||||
										<input type="hidden" name="notification_id" value="{{$notification.ID}}" />
 | 
			
		||||
										<input type="hidden" name="status" value="read" />
 | 
			
		||||
										<button class="ui button" title="Mark as read">
 | 
			
		||||
											<i class="octicon octicon-check"></i>
 | 
			
		||||
										</button>
 | 
			
		||||
									</form>
 | 
			
		||||
								{{else if eq $notification.Status 2}}
 | 
			
		||||
									<form action="{{$.AppSubUrl}}/notifications/status" method="POST">
 | 
			
		||||
										{{$.CsrfTokenHtml}}
 | 
			
		||||
										<input type="hidden" name="notification_id" value="{{$notification.ID}}" />
 | 
			
		||||
										<input type="hidden" name="status" value="unread" />
 | 
			
		||||
										<button class="ui button" title="Mark as unread">
 | 
			
		||||
											<i class="octicon octicon-bell"></i>
 | 
			
		||||
										</button>
 | 
			
		||||
									</form>
 | 
			
		||||
								{{end}}
 | 
			
		||||
							</div>
 | 
			
		||||
 | 
			
		||||
								<div class="content">
 | 
			
		||||
									<div class="header">{{$repoOwner.Name}}/{{$repo.Name}}</div>
 | 
			
		||||
									<div class="description">#{{$issue.Index}} - {{$issue.Title}}</div>
 | 
			
		||||
								</div>
 | 
			
		||||
							</a>
 | 
			
		||||
						</div>
 | 
			
		||||
							{{if eq $notification.Status 3}}
 | 
			
		||||
								<i class="blue octicon octicon-pin"></i>
 | 
			
		||||
							{{else if $issue.IsPull}}
 | 
			
		||||
								{{if $issue.IsClosed}}
 | 
			
		||||
									<i class="octicon octicon-git-merge"></i>
 | 
			
		||||
								{{else}}
 | 
			
		||||
									<i class="octicon octicon-git-pull-request"></i>
 | 
			
		||||
								{{end}}
 | 
			
		||||
							{{else}}
 | 
			
		||||
								{{if $issue.IsClosed}}
 | 
			
		||||
									<i class="octicon octicon-issue-closed"></i>
 | 
			
		||||
								{{else}}
 | 
			
		||||
									<i class="octicon octicon-issue-opened"></i>
 | 
			
		||||
								{{end}}
 | 
			
		||||
							{{end}}
 | 
			
		||||
 | 
			
		||||
							<div class="content">
 | 
			
		||||
								<div class="header">{{$repoOwner.Name}}/{{$repo.Name}}</div>
 | 
			
		||||
								<div class="description">#{{$issue.Index}} - {{$issue.Title}}</div>
 | 
			
		||||
							</div>
 | 
			
		||||
						</a>
 | 
			
		||||
					{{end}}
 | 
			
		||||
				</div>
 | 
			
		||||
			{{end}}
 | 
			
		||||
| 
						 | 
				
			
			
 | 
			
		|||
		Loading…
	
		Reference in New Issue