Blame


1 3baa2617 2022-02-16 op /*
2 3baa2617 2022-02-16 op * Copyright (c) 2022 Omar Polo <op@openbsd.org>
3 3baa2617 2022-02-16 op *
4 3baa2617 2022-02-16 op * Permission to use, copy, modify, and distribute this software for any
5 3baa2617 2022-02-16 op * purpose with or without fee is hereby granted, provided that the above
6 3baa2617 2022-02-16 op * copyright notice and this permission notice appear in all copies.
7 3baa2617 2022-02-16 op *
8 3baa2617 2022-02-16 op * THE SOFTWARE IS PROVIDED "AS IS" AND THE AUTHOR DISCLAIMS ALL WARRANTIES
9 3baa2617 2022-02-16 op * WITH REGARD TO THIS SOFTWARE INCLUDING ALL IMPLIED WARRANTIES OF
10 3baa2617 2022-02-16 op * MERCHANTABILITY AND FITNESS. IN NO EVENT SHALL THE AUTHOR BE LIABLE FOR
11 3baa2617 2022-02-16 op * ANY SPECIAL, DIRECT, INDIRECT, OR CONSEQUENTIAL DAMAGES OR ANY DAMAGES
12 3baa2617 2022-02-16 op * WHATSOEVER RESULTING FROM LOSS OF USE, DATA OR PROFITS, WHETHER IN AN
13 3baa2617 2022-02-16 op * ACTION OF CONTRACT, NEGLIGENCE OR OTHER TORTIOUS ACTION, ARISING OUT OF
14 3baa2617 2022-02-16 op * OR IN CONNECTION WITH THE USE OR PERFORMANCE OF THIS SOFTWARE.
15 3baa2617 2022-02-16 op */
16 3baa2617 2022-02-16 op
17 a913de21 2022-02-17 op #include <sys/types.h>
18 a913de21 2022-02-17 op
19 a913de21 2022-02-17 op #include <regex.h>
20 3baa2617 2022-02-16 op #include <stdlib.h>
21 82e732c9 2022-02-19 op #include <string.h>
22 3baa2617 2022-02-16 op #include <syslog.h>
23 3baa2617 2022-02-16 op
24 3baa2617 2022-02-16 op #include "log.h"
25 3baa2617 2022-02-16 op #include "xmalloc.h"
26 3baa2617 2022-02-16 op #include "playlist.h"
27 3baa2617 2022-02-16 op
28 3baa2617 2022-02-16 op #define MAX(a, b) ((a) > (b) ? (a) : (b))
29 3baa2617 2022-02-16 op
30 74c987d5 2022-02-19 op struct playlist playlist;
31 74c987d5 2022-02-19 op enum play_state play_state;
32 74c987d5 2022-02-19 op int repeat_one;
33 74c987d5 2022-02-19 op int repeat_all = 1;
34 74c987d5 2022-02-19 op ssize_t play_off = -1;
35 74c987d5 2022-02-19 op const char *current_song;
36 3baa2617 2022-02-16 op
37 74c987d5 2022-02-19 op static void
38 74c987d5 2022-02-19 op setsong(ssize_t i)
39 74c987d5 2022-02-19 op {
40 74c987d5 2022-02-19 op free((char *)current_song);
41 74c987d5 2022-02-19 op if (i == -1)
42 74c987d5 2022-02-19 op current_song = NULL;
43 74c987d5 2022-02-19 op else
44 74c987d5 2022-02-19 op current_song = xstrdup(playlist.songs[i]);
45 74c987d5 2022-02-19 op }
46 74c987d5 2022-02-19 op
47 3baa2617 2022-02-16 op void
48 3af93963 2022-03-02 op playlist_swap(struct playlist *p, ssize_t off)
49 0977e96a 2022-02-17 op {
50 82e732c9 2022-02-19 op ssize_t i = -1;
51 82e732c9 2022-02-19 op
52 3af93963 2022-03-02 op if (off > p->len)
53 3af93963 2022-03-02 op off = -1;
54 3af93963 2022-03-02 op
55 3af93963 2022-03-02 op if (current_song != NULL && off < 0) {
56 82e732c9 2022-02-19 op /* try to adjust play_off to match the same song */
57 82e732c9 2022-02-19 op for (i = 0; i < p->len; ++i) {
58 b21ec899 2022-02-19 op if (!strcmp(current_song, p->songs[i]))
59 82e732c9 2022-02-19 op break;
60 82e732c9 2022-02-19 op }
61 82e732c9 2022-02-19 op /* try to match one song before */
62 82e732c9 2022-02-19 op if (i == p->len && play_off >= 1)
63 82e732c9 2022-02-19 op for (i = 0; i < p->len; ++i)
64 b21ec899 2022-02-19 op if (!strcmp(current_song, p->songs[i]))
65 82e732c9 2022-02-19 op break;
66 82e732c9 2022-02-19 op /* or one song after */
67 82e732c9 2022-02-19 op if (i == p->len && play_off < playlist.len-1)
68 82e732c9 2022-02-19 op for (i = 0; i < p->len; ++i)
69 b21ec899 2022-02-19 op if (!strcmp(current_song, p->songs[i]))
70 82e732c9 2022-02-19 op break;
71 82e732c9 2022-02-19 op if (i == p->len)
72 82e732c9 2022-02-19 op i = -1;
73 82e732c9 2022-02-19 op }
74 82e732c9 2022-02-19 op
75 0977e96a 2022-02-17 op playlist_truncate();
76 0977e96a 2022-02-17 op
77 82e732c9 2022-02-19 op if (i != -1)
78 82e732c9 2022-02-19 op play_off = i;
79 3af93963 2022-03-02 op else if (off >= 0)
80 3af93963 2022-03-02 op play_off = off;
81 82e732c9 2022-02-19 op
82 0977e96a 2022-02-17 op playlist.len = p->len;
83 0977e96a 2022-02-17 op playlist.cap = p->cap;
84 0977e96a 2022-02-17 op playlist.songs = p->songs;
85 76079111 2022-03-02 op
86 76079111 2022-03-02 op if (play_state == STATE_STOPPED)
87 76079111 2022-03-02 op setsong(play_off);
88 0977e96a 2022-02-17 op }
89 0977e96a 2022-02-17 op
90 0977e96a 2022-02-17 op void
91 532ca63c 2022-02-17 op playlist_push(struct playlist *playlist, const char *path)
92 3baa2617 2022-02-16 op {
93 3baa2617 2022-02-16 op size_t newcap;
94 3baa2617 2022-02-16 op
95 532ca63c 2022-02-17 op if (playlist->len == playlist->cap) {
96 532ca63c 2022-02-17 op newcap = MAX(16, playlist->cap * 1.5);
97 532ca63c 2022-02-17 op playlist->songs = xrecallocarray(playlist->songs,
98 532ca63c 2022-02-17 op playlist->cap, newcap, sizeof(*playlist->songs));
99 532ca63c 2022-02-17 op playlist->cap = newcap;
100 3baa2617 2022-02-16 op }
101 3baa2617 2022-02-16 op
102 532ca63c 2022-02-17 op playlist->songs[playlist->len++] = xstrdup(path);
103 3baa2617 2022-02-16 op }
104 3baa2617 2022-02-16 op
105 532ca63c 2022-02-17 op void
106 532ca63c 2022-02-17 op playlist_enqueue(const char *path)
107 532ca63c 2022-02-17 op {
108 532ca63c 2022-02-17 op playlist_push(&playlist, path);
109 532ca63c 2022-02-17 op }
110 532ca63c 2022-02-17 op
111 3baa2617 2022-02-16 op const char *
112 3baa2617 2022-02-16 op playlist_advance(void)
113 3baa2617 2022-02-16 op {
114 8891f624 2022-02-16 op if (playlist.len == 0) {
115 8891f624 2022-02-16 op play_state = STATE_STOPPED;
116 3baa2617 2022-02-16 op return NULL;
117 8891f624 2022-02-16 op }
118 3baa2617 2022-02-16 op
119 3baa2617 2022-02-16 op play_off++;
120 3baa2617 2022-02-16 op if (play_off == playlist.len) {
121 3baa2617 2022-02-16 op if (repeat_all)
122 3baa2617 2022-02-16 op play_off = 0;
123 3baa2617 2022-02-16 op else {
124 3baa2617 2022-02-16 op play_state = STATE_STOPPED;
125 3baa2617 2022-02-16 op play_off = -1;
126 74c987d5 2022-02-19 op setsong(play_off);
127 3baa2617 2022-02-16 op return NULL;
128 3baa2617 2022-02-16 op }
129 3baa2617 2022-02-16 op }
130 3baa2617 2022-02-16 op
131 74c987d5 2022-02-19 op setsong(play_off);
132 3baa2617 2022-02-16 op play_state = STATE_PLAYING;
133 3baa2617 2022-02-16 op return playlist.songs[play_off];
134 3baa2617 2022-02-16 op }
135 3baa2617 2022-02-16 op
136 af27e631 2022-02-17 op const char *
137 af27e631 2022-02-17 op playlist_previous(void)
138 af27e631 2022-02-17 op {
139 af27e631 2022-02-17 op if (playlist.len == 0) {
140 af27e631 2022-02-17 op play_state = STATE_STOPPED;
141 af27e631 2022-02-17 op return NULL;
142 af27e631 2022-02-17 op }
143 af27e631 2022-02-17 op
144 af27e631 2022-02-17 op play_off--;
145 af27e631 2022-02-17 op if (play_off < 0) {
146 af27e631 2022-02-17 op if (repeat_all)
147 af27e631 2022-02-17 op play_off = playlist.len - 1;
148 af27e631 2022-02-17 op else {
149 af27e631 2022-02-17 op play_state = STATE_STOPPED;
150 af27e631 2022-02-17 op play_off = -1;
151 74c987d5 2022-02-19 op setsong(play_off);
152 af27e631 2022-02-17 op return NULL;
153 af27e631 2022-02-17 op }
154 af27e631 2022-02-17 op }
155 af27e631 2022-02-17 op
156 74c987d5 2022-02-19 op setsong(play_off);
157 af27e631 2022-02-17 op play_state = STATE_PLAYING;
158 af27e631 2022-02-17 op return playlist.songs[play_off];
159 af27e631 2022-02-17 op }
160 af27e631 2022-02-17 op
161 3baa2617 2022-02-16 op void
162 3baa2617 2022-02-16 op playlist_reset(void)
163 3baa2617 2022-02-16 op {
164 3baa2617 2022-02-16 op play_off = -1;
165 3baa2617 2022-02-16 op }
166 3baa2617 2022-02-16 op
167 3baa2617 2022-02-16 op void
168 cd070aea 2022-02-17 op playlist_free(struct playlist *playlist)
169 3baa2617 2022-02-16 op {
170 3baa2617 2022-02-16 op size_t i;
171 3baa2617 2022-02-16 op
172 cd070aea 2022-02-17 op for (i = 0; i < playlist->len; ++i)
173 cd070aea 2022-02-17 op free(playlist->songs[i]);
174 cd070aea 2022-02-17 op free(playlist->songs);
175 cd070aea 2022-02-17 op playlist->songs = NULL;
176 3baa2617 2022-02-16 op
177 cd070aea 2022-02-17 op playlist->len = 0;
178 cd070aea 2022-02-17 op playlist->cap = 0;
179 cd070aea 2022-02-17 op }
180 cd070aea 2022-02-17 op
181 cd070aea 2022-02-17 op void
182 cd070aea 2022-02-17 op playlist_truncate(void)
183 cd070aea 2022-02-17 op {
184 cd070aea 2022-02-17 op playlist_free(&playlist);
185 3baa2617 2022-02-16 op play_off = -1;
186 3baa2617 2022-02-16 op }
187 13b83883 2022-02-16 op
188 13b83883 2022-02-16 op void
189 13b83883 2022-02-16 op playlist_dropcurrent(void)
190 13b83883 2022-02-16 op {
191 13b83883 2022-02-16 op size_t i;
192 13b83883 2022-02-16 op
193 13b83883 2022-02-16 op if (play_off == -1 || playlist.len == 0)
194 13b83883 2022-02-16 op return;
195 13b83883 2022-02-16 op
196 13b83883 2022-02-16 op free(playlist.songs[play_off]);
197 601093db 2022-02-22 op setsong(-1);
198 13b83883 2022-02-16 op
199 13b83883 2022-02-16 op playlist.len--;
200 13b83883 2022-02-16 op for (i = play_off; i < playlist.len; ++i)
201 13b83883 2022-02-16 op playlist.songs[i] = playlist.songs[i+1];
202 a553fbed 2022-02-22 op play_off--;
203 13b83883 2022-02-16 op
204 13b83883 2022-02-16 op playlist.songs[playlist.len] = NULL;
205 13b83883 2022-02-16 op }
206 a913de21 2022-02-17 op
207 a913de21 2022-02-17 op const char *
208 a913de21 2022-02-17 op playlist_jump(const char *arg)
209 a913de21 2022-02-17 op {
210 a913de21 2022-02-17 op size_t i;
211 a913de21 2022-02-17 op regex_t re;
212 a913de21 2022-02-17 op
213 a913de21 2022-02-17 op if (regcomp(&re, arg, REG_ICASE | REG_NOSUB) != 0)
214 a913de21 2022-02-17 op return NULL;
215 a913de21 2022-02-17 op
216 a913de21 2022-02-17 op for (i = 0; i < playlist.len; ++i) {
217 a913de21 2022-02-17 op if (regexec(&re, playlist.songs[i], 0, NULL, 0) == 0)
218 a913de21 2022-02-17 op break;
219 a913de21 2022-02-17 op }
220 a913de21 2022-02-17 op regfree(&re);
221 a913de21 2022-02-17 op
222 a913de21 2022-02-17 op if (i == playlist.len)
223 a913de21 2022-02-17 op return NULL;
224 a913de21 2022-02-17 op
225 a913de21 2022-02-17 op play_state = STATE_PLAYING;
226 a913de21 2022-02-17 op play_off = i;
227 74c987d5 2022-02-19 op setsong(play_off);
228 a913de21 2022-02-17 op return playlist.songs[i];
229 a913de21 2022-02-17 op }