2019-12-11 16:55:18 +01:00
|
|
|
package entity
|
2018-07-18 15:17:56 +02:00
|
|
|
|
|
|
|
import (
|
2019-12-08 15:05:35 +01:00
|
|
|
"sort"
|
2018-08-09 23:10:05 +02:00
|
|
|
"time"
|
2018-10-31 07:14:33 +01:00
|
|
|
|
|
|
|
"github.com/jinzhu/gorm"
|
2019-12-06 10:26:57 +01:00
|
|
|
"github.com/photoprism/photoprism/internal/util"
|
2019-06-09 05:22:53 +02:00
|
|
|
uuid "github.com/satori/go.uuid"
|
2018-07-18 15:17:56 +02:00
|
|
|
)
|
|
|
|
|
2018-11-06 19:02:03 +01:00
|
|
|
// A photo can have multiple images and sidecar files
|
2018-07-18 15:17:56 +02:00
|
|
|
type Photo struct {
|
2019-06-04 18:26:35 +02:00
|
|
|
Model
|
2019-07-02 18:25:46 +02:00
|
|
|
PhotoUUID string `gorm:"unique_index;"`
|
2019-12-06 10:26:57 +01:00
|
|
|
PhotoToken string `gorm:"type:varchar(64);"`
|
2019-07-02 18:25:46 +02:00
|
|
|
PhotoPath string `gorm:"type:varchar(128);index;"`
|
|
|
|
PhotoName string
|
2019-12-11 19:11:44 +01:00
|
|
|
PhotoTitle string `json:"PhotoTitle"`
|
2019-07-02 18:25:46 +02:00
|
|
|
PhotoTitleChanged bool
|
2019-12-11 19:11:44 +01:00
|
|
|
PhotoDescription string `gorm:"type:text;"`
|
|
|
|
PhotoNotes string `gorm:"type:text;"`
|
|
|
|
PhotoArtist string `json:"PhotoArtist"`
|
|
|
|
PhotoFavorite bool `json:"PhotoFavorite"`
|
|
|
|
PhotoPrivate bool `json:"PhotoPrivate"`
|
|
|
|
PhotoNSFW bool `json:"PhotoNSFW"`
|
|
|
|
PhotoStory bool `json:"PhotoStory"`
|
2019-07-02 22:14:20 +02:00
|
|
|
PhotoLat float64 `gorm:"index;"`
|
|
|
|
PhotoLong float64 `gorm:"index;"`
|
2019-07-02 18:25:46 +02:00
|
|
|
PhotoAltitude int
|
|
|
|
PhotoFocalLength int
|
|
|
|
PhotoIso int
|
2019-11-07 19:00:26 +01:00
|
|
|
PhotoFNumber float64
|
2019-07-02 18:25:46 +02:00
|
|
|
PhotoExposure string
|
|
|
|
PhotoViews uint
|
|
|
|
Camera *Camera
|
2019-07-15 22:54:54 +02:00
|
|
|
CameraID uint `gorm:"index;"`
|
2019-07-02 18:25:46 +02:00
|
|
|
Lens *Lens
|
2019-07-15 22:54:54 +02:00
|
|
|
LensID uint `gorm:"index;"`
|
2019-07-02 18:25:46 +02:00
|
|
|
Country *Country
|
2019-07-15 22:54:54 +02:00
|
|
|
CountryID string `gorm:"index;"`
|
2019-07-02 18:25:46 +02:00
|
|
|
CountryChanged bool
|
|
|
|
Location *Location
|
|
|
|
LocationID uint
|
|
|
|
LocationChanged bool
|
|
|
|
LocationEstimated bool
|
2019-07-02 22:14:20 +02:00
|
|
|
TakenAt time.Time `gorm:"index;"`
|
2019-09-19 23:23:39 +02:00
|
|
|
TakenAtLocal time.Time
|
2019-07-02 18:25:46 +02:00
|
|
|
TakenAtChanged bool
|
|
|
|
TimeZone string
|
2019-12-11 19:11:44 +01:00
|
|
|
Files []File
|
|
|
|
Labels []Label
|
|
|
|
Keywords []Keyword
|
|
|
|
Albums []Album
|
2018-08-15 09:59:51 +02:00
|
|
|
}
|
2019-06-04 18:26:35 +02:00
|
|
|
|
|
|
|
func (m *Photo) BeforeCreate(scope *gorm.Scope) error {
|
2019-12-06 10:26:57 +01:00
|
|
|
if err := scope.SetColumn("PhotoUUID", uuid.NewV4().String()); err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
|
|
|
if err := scope.SetColumn("PhotoToken", util.RandomToken(4)); err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
|
|
|
return nil
|
2019-06-04 18:26:35 +02:00
|
|
|
}
|
2019-12-08 15:05:35 +01:00
|
|
|
|
|
|
|
func (m *Photo) IndexKeywords(keywords []string, db *gorm.DB) {
|
|
|
|
var keywordIds []uint
|
|
|
|
|
|
|
|
// Index title and description
|
|
|
|
keywords = append(keywords, util.Keywords(m.PhotoTitle)...)
|
|
|
|
keywords = append(keywords, util.Keywords(m.PhotoDescription)...)
|
|
|
|
last := ""
|
|
|
|
|
|
|
|
sort.Strings(keywords)
|
|
|
|
|
|
|
|
for _, w := range keywords {
|
|
|
|
if len(w) < 3 || w == last {
|
|
|
|
continue
|
|
|
|
}
|
|
|
|
|
|
|
|
last = w
|
|
|
|
kw := NewKeyword(w).FirstOrCreate(db)
|
|
|
|
|
|
|
|
if kw.Skip {
|
|
|
|
continue
|
|
|
|
}
|
|
|
|
|
|
|
|
keywordIds = append(keywordIds, kw.ID)
|
|
|
|
|
|
|
|
NewPhotoKeyword(m.ID, kw.ID).FirstOrCreate(db)
|
|
|
|
}
|
|
|
|
|
|
|
|
db.Where("photo_id = ? AND keyword_id NOT IN (?)", m.ID, keywordIds).Delete(&PhotoKeyword{})
|
|
|
|
}
|
2019-12-11 19:11:44 +01:00
|
|
|
|
|
|
|
func (m *Photo) PreloadFiles(db *gorm.DB) {
|
|
|
|
q := db.NewScope(nil).DB().
|
|
|
|
Table("files").
|
|
|
|
Select(`files.*`).
|
|
|
|
Where("files.photo_id = ?", m.ID).
|
|
|
|
Order("files.file_primary DESC")
|
|
|
|
|
|
|
|
logError(q.Scan(&m.Files))
|
|
|
|
}
|
|
|
|
|
|
|
|
func (m *Photo) PreloadLabels(db *gorm.DB) {
|
|
|
|
q := db.NewScope(nil).DB().
|
|
|
|
Table("labels").
|
|
|
|
Select(`labels.*`).
|
|
|
|
Joins("JOIN photos_labels ON photos_labels.label_id = labels.id AND photos_labels.photo_id = ?", m.ID).
|
|
|
|
Where("labels.deleted_at IS NULL").
|
|
|
|
Order("labels.label_name ASC")
|
|
|
|
|
|
|
|
logError(q.Scan(&m.Labels))
|
|
|
|
}
|
|
|
|
|
|
|
|
func (m *Photo) PreloadKeywords(db *gorm.DB) {
|
|
|
|
q := db.NewScope(nil).DB().
|
|
|
|
Table("keywords").
|
|
|
|
Select(`keywords.*`).
|
|
|
|
Joins("JOIN photos_keywords ON photos_keywords.keyword_id = keywords.id AND photos_keywords.photo_id = ?", m.ID).
|
|
|
|
Order("keywords.keyword ASC")
|
|
|
|
|
|
|
|
logError(q.Scan(&m.Keywords))
|
|
|
|
}
|
|
|
|
|
|
|
|
func (m *Photo) PreloadAlbums(db *gorm.DB) {
|
|
|
|
q := db.NewScope(nil).DB().
|
|
|
|
Table("albums").
|
|
|
|
Select(`albums.*`).
|
|
|
|
Joins("JOIN photos_albums ON photos_albums.album_uuid = albums.album_uuid AND photos_albums.photo_uuid = ?", m.PhotoUUID).
|
|
|
|
Where("albums.deleted_at IS NULL").
|
|
|
|
Order("albums.album_name ASC")
|
|
|
|
|
|
|
|
logError(q.Scan(&m.Albums))
|
|
|
|
}
|
|
|
|
|
|
|
|
func (m *Photo) PreloadMany(db *gorm.DB) {
|
|
|
|
m.PreloadFiles(db)
|
|
|
|
m.PreloadLabels(db)
|
|
|
|
m.PreloadKeywords(db)
|
|
|
|
m.PreloadAlbums(db)
|
|
|
|
}
|