2019-12-28 20:24:20 +01:00
|
|
|
package entity
|
|
|
|
|
|
|
|
import (
|
2020-05-26 21:51:34 +02:00
|
|
|
"strings"
|
2019-12-28 20:24:20 +01:00
|
|
|
"time"
|
|
|
|
|
|
|
|
"github.com/jinzhu/gorm"
|
|
|
|
"github.com/photoprism/photoprism/internal/maps"
|
2020-05-03 18:00:50 +02:00
|
|
|
"github.com/photoprism/photoprism/pkg/txt"
|
2019-12-28 20:24:20 +01:00
|
|
|
)
|
|
|
|
|
2020-02-21 01:14:45 +01:00
|
|
|
// Place used to associate photos to places
|
2019-12-28 20:24:20 +01:00
|
|
|
type Place struct {
|
2020-05-25 19:10:44 +02:00
|
|
|
PlaceUID string `gorm:"type:varbinary(16);primary_key;auto_increment:false;" json:"PlaceUID" yaml:"PlaceUID"`
|
|
|
|
LocLabel string `gorm:"type:varbinary(768);unique_index;" json:"Label" yaml:"Label"`
|
|
|
|
LocCity string `gorm:"type:varchar(255);" json:"City" yaml:"City,omitempty"`
|
|
|
|
LocState string `gorm:"type:varchar(255);" json:"State" yaml:"State,omitempty"`
|
|
|
|
LocCountry string `gorm:"type:varbinary(2);" json:"Country" yaml:"Country,omitempty"`
|
|
|
|
LocKeywords string `gorm:"type:varchar(255);" json:"Keywords" yaml:"Keywords,omitempty"`
|
|
|
|
LocNotes string `gorm:"type:text;" json:"Notes" yaml:"Notes,omitempty"`
|
|
|
|
LocFavorite bool `json:"Favorite" yaml:"Favorite,omitempty"`
|
|
|
|
PhotoCount int `json:"PhotoCount" yaml:"-"`
|
|
|
|
CreatedAt time.Time `json:"CreatedAt" yaml:"-"`
|
|
|
|
UpdatedAt time.Time `json:"UpdatedAt" yaml:"-"`
|
|
|
|
New bool `gorm:"-" json:"-" yaml:"-"`
|
2019-12-28 20:24:20 +01:00
|
|
|
}
|
|
|
|
|
2020-05-25 19:10:44 +02:00
|
|
|
// UnknownPlace is PhotoPrism's default place.
|
2020-04-25 14:22:47 +02:00
|
|
|
var UnknownPlace = Place{
|
2020-05-25 19:10:44 +02:00
|
|
|
PlaceUID: "zz",
|
2020-04-28 19:41:06 +02:00
|
|
|
LocLabel: "Unknown",
|
|
|
|
LocCity: "Unknown",
|
|
|
|
LocState: "Unknown",
|
|
|
|
LocCountry: "zz",
|
|
|
|
LocKeywords: "",
|
|
|
|
LocNotes: "",
|
|
|
|
LocFavorite: false,
|
2020-05-08 15:41:01 +02:00
|
|
|
PhotoCount: -1,
|
2020-04-25 14:22:47 +02:00
|
|
|
}
|
2019-12-28 20:24:20 +01:00
|
|
|
|
2020-05-25 19:10:44 +02:00
|
|
|
// CreateUnknownPlace creates the default place if not exists.
|
2020-04-30 20:07:03 +02:00
|
|
|
func CreateUnknownPlace() {
|
2020-05-25 19:10:44 +02:00
|
|
|
FirstOrCreatePlace(&UnknownPlace)
|
2019-12-28 20:24:20 +01:00
|
|
|
}
|
|
|
|
|
2020-02-21 01:14:45 +01:00
|
|
|
// AfterCreate sets the New column used for database callback
|
2019-12-28 20:24:20 +01:00
|
|
|
func (m *Place) AfterCreate(scope *gorm.Scope) error {
|
2020-05-19 12:30:26 +02:00
|
|
|
m.New = true
|
|
|
|
return nil
|
2019-12-28 20:24:20 +01:00
|
|
|
}
|
|
|
|
|
2020-02-21 01:14:45 +01:00
|
|
|
// FindPlaceByLabel returns a place from an id or a label
|
2020-05-25 19:10:44 +02:00
|
|
|
func FindPlaceByLabel(uid string, label string) *Place {
|
2019-12-28 20:24:20 +01:00
|
|
|
place := &Place{}
|
|
|
|
|
2020-05-15 13:16:03 +02:00
|
|
|
if label == "" {
|
2020-05-25 19:10:44 +02:00
|
|
|
if err := Db().First(place, "place_uid = ?", uid).Error; err != nil {
|
|
|
|
log.Debugf("place: %s for uid %s", err.Error(), uid)
|
2020-05-15 13:16:03 +02:00
|
|
|
return nil
|
|
|
|
}
|
2020-05-25 19:10:44 +02:00
|
|
|
} else if err := Db().First(place, "place_uid = ? OR loc_label = ?", uid, label).Error; err != nil {
|
|
|
|
log.Debugf("place: %s for uid %s / label %s", err.Error(), uid, txt.Quote(label))
|
2020-04-25 16:17:59 +02:00
|
|
|
return nil
|
2019-12-28 20:24:20 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
return place
|
|
|
|
}
|
|
|
|
|
2020-05-26 11:00:39 +02:00
|
|
|
// Find updates the entity with values from the database.
|
2020-04-30 20:07:03 +02:00
|
|
|
func (m *Place) Find() error {
|
2020-05-25 19:10:44 +02:00
|
|
|
if err := Db().First(m, "place_uid = ?", m.PlaceUID).Error; err != nil {
|
2019-12-28 20:24:20 +01:00
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
2020-05-25 19:10:44 +02:00
|
|
|
// Create inserts a new row to the database.
|
|
|
|
func (m *Place) Create() error {
|
2020-05-26 11:00:39 +02:00
|
|
|
return Db().Create(m).Error
|
2020-05-25 19:10:44 +02:00
|
|
|
}
|
|
|
|
|
2020-05-26 11:00:39 +02:00
|
|
|
// FirstOrCreatePlace returns the existing row, inserts a new row or nil in case of errors.
|
2020-05-25 19:10:44 +02:00
|
|
|
func FirstOrCreatePlace(m *Place) *Place {
|
|
|
|
if m.PlaceUID == "" {
|
|
|
|
log.Errorf("place: uid must not be empty")
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
|
|
|
if m.LocLabel == "" {
|
|
|
|
log.Errorf("place: label must not be empty (uid %s)", m.PlaceUID)
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
|
|
|
result := Place{}
|
|
|
|
|
|
|
|
if err := Db().Where("place_uid = ? OR loc_label = ?", m.PlaceUID, m.LocLabel).First(&result).Error; err == nil {
|
|
|
|
return &result
|
|
|
|
} else if err := m.Create(); err != nil {
|
|
|
|
log.Errorf("place: %s", err)
|
|
|
|
return nil
|
2019-12-28 20:24:20 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
return m
|
|
|
|
}
|
|
|
|
|
2020-04-25 16:17:59 +02:00
|
|
|
// Unknown returns true if this is an unknown place
|
|
|
|
func (m Place) Unknown() bool {
|
2020-05-26 21:51:34 +02:00
|
|
|
return m.PlaceUID == "" || m.PlaceUID == UnknownPlace.PlaceUID
|
2019-12-28 20:24:20 +01:00
|
|
|
}
|
|
|
|
|
2020-02-21 01:14:45 +01:00
|
|
|
// Label returns place label
|
2020-04-16 20:57:00 +02:00
|
|
|
func (m Place) Label() string {
|
2019-12-28 20:24:20 +01:00
|
|
|
return m.LocLabel
|
|
|
|
}
|
|
|
|
|
2020-02-21 01:14:45 +01:00
|
|
|
// City returns place City
|
2020-04-16 20:57:00 +02:00
|
|
|
func (m Place) City() string {
|
2019-12-28 20:24:20 +01:00
|
|
|
return m.LocCity
|
|
|
|
}
|
|
|
|
|
2020-05-26 21:51:34 +02:00
|
|
|
// LongCity checks if the city name is more than 16 char.
|
|
|
|
func (m Place) LongCity() bool {
|
|
|
|
return len(m.LocCity) > 16
|
|
|
|
}
|
|
|
|
|
|
|
|
// NoCity checks if the location has no city
|
|
|
|
func (m Place) NoCity() bool {
|
|
|
|
return m.LocCity == ""
|
|
|
|
}
|
|
|
|
|
|
|
|
// CityContains checks if the location city contains the text string
|
|
|
|
func (m Place) CityContains(text string) bool {
|
|
|
|
return strings.Contains(text, m.LocCity)
|
|
|
|
}
|
|
|
|
|
2020-02-21 01:14:45 +01:00
|
|
|
// State returns place State
|
2020-04-16 20:57:00 +02:00
|
|
|
func (m Place) State() string {
|
2019-12-28 20:24:20 +01:00
|
|
|
return m.LocState
|
|
|
|
}
|
|
|
|
|
2020-02-21 01:14:45 +01:00
|
|
|
// CountryCode returns place CountryCode
|
2020-04-16 20:57:00 +02:00
|
|
|
func (m Place) CountryCode() string {
|
2019-12-28 20:24:20 +01:00
|
|
|
return m.LocCountry
|
|
|
|
}
|
|
|
|
|
2020-02-21 01:14:45 +01:00
|
|
|
// CountryName returns place CountryName
|
2020-04-16 20:57:00 +02:00
|
|
|
func (m Place) CountryName() string {
|
2019-12-28 20:24:20 +01:00
|
|
|
return maps.CountryNames[m.LocCountry]
|
|
|
|
}
|
|
|
|
|
2020-02-21 01:14:45 +01:00
|
|
|
// Notes returns place Notes
|
2020-04-16 20:57:00 +02:00
|
|
|
func (m Place) Notes() string {
|
2019-12-28 20:24:20 +01:00
|
|
|
return m.LocNotes
|
|
|
|
}
|