This repository has been archived by the owner on Jun 3, 2019. It is now read-only.
-
Notifications
You must be signed in to change notification settings - Fork 3
/
Copy pathAnimeSort.go
107 lines (87 loc) · 2.86 KB
/
AnimeSort.go
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
package arn
import (
"fmt"
"sort"
"time"
)
const (
currentlyAiringBonus = 5.0
longSummaryBonus = 0.1
popularityThreshold = 5
popularityPenalty = 8.0
watchingPopularityWeight = 0.07
completedPopularityWeight = watchingPopularityWeight
plannedPopularityWeight = watchingPopularityWeight * (2.0 / 3.0)
droppedPopularityWeight = -plannedPopularityWeight
visualsWeight = 0.0075
storyWeight = 0.0075
soundtrackWeight = 0.0075
movieBonus = 0.28
agePenalty = 11.0
ageThreshold = 6 * 30 * 24 * time.Hour
)
// SortAnimeByPopularity sorts the given slice of anime by popularity.
func SortAnimeByPopularity(animes []*Anime) {
sort.Slice(animes, func(i, j int) bool {
aPopularity := animes[i].Popularity.Total()
bPopularity := animes[j].Popularity.Total()
if aPopularity == bPopularity {
return animes[i].Title.Canonical < animes[j].Title.Canonical
}
return aPopularity > bPopularity
})
}
// SortAnimeByQuality sorts the given slice of anime by quality.
func SortAnimeByQuality(animes []*Anime) {
SortAnimeByQualityDetailed(animes, "")
}
// SortAnimeByQualityDetailed sorts the given slice of anime by quality.
func SortAnimeByQualityDetailed(animes []*Anime, filterStatus string) {
sort.Slice(animes, func(i, j int) bool {
a := animes[i]
b := animes[j]
scoreA := a.Score()
scoreB := b.Score()
// If we show currently running shows, rank shows that started a long time ago a bit lower
if filterStatus == "current" {
if a.StartDate != "" && time.Since(a.StartDateTime()) > ageThreshold {
scoreA -= agePenalty
}
if b.StartDate != "" && time.Since(b.StartDateTime()) > ageThreshold {
scoreB -= agePenalty
}
}
if scoreA == scoreB {
return a.Title.Canonical < b.Title.Canonical
}
return scoreA > scoreB
})
}
// Score returns the score used for the anime ranking.
func (anime *Anime) Score() float64 {
score := anime.Rating.Overall
score += anime.Rating.Story * storyWeight
score += anime.Rating.Visuals * visualsWeight
score += anime.Rating.Soundtrack * soundtrackWeight
score += float64(anime.Popularity.Watching) * watchingPopularityWeight
score += float64(anime.Popularity.Planned) * plannedPopularityWeight
score += float64(anime.Popularity.Completed) * completedPopularityWeight
score += float64(anime.Popularity.Dropped) * droppedPopularityWeight
if anime.Status == "current" {
score += currentlyAiringBonus
}
if anime.Type == "movie" {
score += movieBonus
}
if anime.Popularity.Total() < popularityThreshold {
score -= popularityPenalty
}
if len(anime.Summary) >= 140 {
score += longSummaryBonus
}
return score
}
// ScoreHumanReadable returns the score used for the anime ranking in human readable format.
func (anime *Anime) ScoreHumanReadable() string {
return fmt.Sprintf("%.1f", anime.Score())
}