-
Notifications
You must be signed in to change notification settings - Fork 4
/
Copy pathsoccer.el
704 lines (647 loc) · 31.7 KB
/
soccer.el
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
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
297
298
299
300
301
302
303
304
305
306
307
308
309
310
311
312
313
314
315
316
317
318
319
320
321
322
323
324
325
326
327
328
329
330
331
332
333
334
335
336
337
338
339
340
341
342
343
344
345
346
347
348
349
350
351
352
353
354
355
356
357
358
359
360
361
362
363
364
365
366
367
368
369
370
371
372
373
374
375
376
377
378
379
380
381
382
383
384
385
386
387
388
389
390
391
392
393
394
395
396
397
398
399
400
401
402
403
404
405
406
407
408
409
410
411
412
413
414
415
416
417
418
419
420
421
422
423
424
425
426
427
428
429
430
431
432
433
434
435
436
437
438
439
440
441
442
443
444
445
446
447
448
449
450
451
452
453
454
455
456
457
458
459
460
461
462
463
464
465
466
467
468
469
470
471
472
473
474
475
476
477
478
479
480
481
482
483
484
485
486
487
488
489
490
491
492
493
494
495
496
497
498
499
500
501
502
503
504
505
506
507
508
509
510
511
512
513
514
515
516
517
518
519
520
521
522
523
524
525
526
527
528
529
530
531
532
533
534
535
536
537
538
539
540
541
542
543
544
545
546
547
548
549
550
551
552
553
554
555
556
557
558
559
560
561
562
563
564
565
566
567
568
569
570
571
572
573
574
575
576
577
578
579
580
581
582
583
584
585
586
587
588
589
590
591
592
593
594
595
596
597
598
599
600
601
602
603
604
605
606
607
608
609
610
611
612
613
614
615
616
617
618
619
620
621
622
623
624
625
626
627
628
629
630
631
632
633
634
635
636
637
638
639
640
641
642
643
644
645
646
647
648
649
650
651
652
653
654
655
656
657
658
659
660
661
662
663
664
665
666
667
668
669
670
671
672
673
674
675
676
677
678
679
680
681
682
683
684
685
686
687
688
689
690
691
692
693
694
695
696
697
698
699
700
701
702
703
704
;;; soccer.el --- Fixtures, results, table etc for soccer -*- lexical-binding: t; -*-
;; Copyright (C) 2021 Md Arif Shaikh
;; Author: Md Arif Shaikh <[email protected]>
;; Homepage: https://github.com/md-arif-shaikh/soccer
;; Package-Requires: ((emacs "26.1") (dash "2.19.1"))
;; Version: 1.2.0
;; Keywords: games, soccer, football
;; This program is free software; you can redistribute it and/or modify
;; it under the terms of the GNU General Public License as published by
;; the Free Software Foundation, either version 3 of the License, or
;; (at your option) any later version.
;; This program is distributed in the hope that it will be useful,
;; but WITHOUT ANY WARRANTY; without even the implied warranty of
;; MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
;; GNU General Public License for more details.
;; You should have received a copy of the GNU General Public License
;; along with this program. If not, see <https://www.gnu.org/licenses/>.
;;; Commentary:
;;
;; This package brings soccer (football) fixtures, results, table in your Emacs.
;; Currently it works for
;;
;; 1. Premier League (England)
;; 2. La Liga (Spain)
;; 3. Ligue 1 (France)
;;
;; Other leagues could be easily included and would be in future.
;;
;; To get the time of kick off in your local time, you may want to set the
;; following value accordingly
;;
;; (setq soccer-time-local-time-utc-offset "+0530") ;; this should be changed to your local one
;;
;; Common Functions:
;;
;; This package comes with handy interactive functions to get useful information in your minibuffer.
;; To invoke a function use "M-x Function" where Function could be any of the following functions
;;
;; Functions Actions
;; soccer-fixtures-next Fixture for the Next match
;; soccer-fixtures-next-5 Fixtures of the Next 5 matches
;; soccer-fixtures-full-in-org Full fixtures saved in org file
;; soccer-fixtures-all-clubs Fixtures for all clubs in a league
;; soccer-results-last Result of the last match
;; soccer-results-last-5 Results of the last 5 matches
;; soccer-results-full-in-org Full list of results in org file
;; soccer-results-all-clubs Results for all clubs in a league
;; soccer-table Full Ranking table
;; soccer-table-top-4 Ranking table with top 4 teams
;; soccer-table-bottom-4 Ranking table with bottom 4 teams
;; soccer-scorecard Scorecard of a match
;;; Code:
(require 'soccer-leagues)
(require 'soccer-time)
(require 'org)
(require 'dom)
(require 'dash)
(defvar soccer-color--win "#98C379"
"Color to indicate a win.")
(defvar soccer-color--loss "#BE5046"
"Color to indicate a loss.")
(defvar soccer-color--draw "#56B6C2"
"Color to indicate a draw.")
(defvar soccer-color--fixtures "#C678DD"
"Color to indicate a fixture.")
(defvar soccer-color--time-to-kickoff "#E5C07B"
"Color to indicate time to kickoff.")
(defvar soccer-color--scorecard-header "#E5C07B"
"Color for scorecard title.")
(defvar soccer-color--local-time "#E5C07B"
"Color for local time.")
(defface soccer-face-local-time
`((t :foreground ,soccer-color--local-time
:weight extra-bold
:box nil
:underline nil))
"Face for local time."
:group 'soccer-face)
(defface soccer-face-win
`((t :foreground ,soccer-color--win
:weight extra-bold
:box nil
:underline nil))
"Face for win."
:group 'soccer-face)
(defface soccer-face-loss
`((t :foreground ,soccer-color--loss
:weight extra-bold
:box nil
:underline nil))
"Face for loss."
:group 'soccer-face)
(defface soccer-face-draw
`((t :foreground ,soccer-color--draw
:weight extra-bold
:box nil
:underline nil))
"Face for draw."
:group 'soccer-face)
(defface soccer-face-fixtures
`((t :foreground ,soccer-color--fixtures
:weight extra-bold
:box nil
:underline nil))
"Face for fixtures."
:group 'soccer-face)
(defface soccer-face-time-to-kickoff
`((t :foreground ,soccer-color--time-to-kickoff
:weight extra-bold
:box nil
:underline nil))
"Face for time to kickoff."
:group 'soccer-face)
(defface soccer-face-scorecard-header
`((t :foreground ,soccer-color--scorecard-header
:weight extra-bold
:box nil
:underline t))
"Face for soccer scorecard title."
:group 'soccer-face)
(defun soccer--get-league-names ()
"Extract league names from `soccer-leagues--leagues-alist'."
(mapcar 'car soccer-leagues--leagues-alist))
(defun soccer--get-league-url (league)
"Get url of a LEAGUE."
(soccer-leagues--get-base-league-url league))
(defun soccer--get-date-from-datetime (datetime)
"Get date from DATETIME string."
(if datetime
(nth 0 (split-string datetime "T"))
""))
(defun soccer--get-time-and-offset-from-datetime (datetime)
"Get time from DATETIME string."
(if datetime
(let* ((timestring (nth 1 (split-string datetime "T"))))
(list (substring timestring 0 -8) (substring timestring -5)))
""))
(defun soccer--get-league-data-alist (league data-type &optional club)
"Get data of DATA-TYPE for a LEAGUE. Optionally filter by CLUB."
(let* ((url (concat (soccer--get-league-url league) "/" (downcase data-type))))
(with-current-buffer (url-retrieve-synchronously url)
(let* ((dom (libxml-parse-html-region (point-min) (point-max)))
(datetimes-dom (dom-by-tag dom 'time))
(homes-dom (dom-by-class dom "football-match__team football-match__team--home football-team"))
(aways-dom (dom-by-class dom "football-match__team football-match__team--away football-team"))
datetimes
dates
times
utc-offsets
homes
aways
results
league-data)
(when (string-equal data-type "fixtures")
(setq datetimes (cl-loop for datetime in datetimes-dom
collect (dom-attr datetime 'datetime)))
(setq dates (cl-loop for datetime in datetimes
collect (soccer--get-date-from-datetime datetime)))
(setq times (cl-loop for datetime in datetimes
collect (nth 0 (soccer--get-time-and-offset-from-datetime datetime))))
(setq utc-offsets (cl-loop for datetime in datetimes
collect (nth 1 (soccer--get-time-and-offset-from-datetime datetime)))))
(setq homes (cl-loop for d in homes-dom
collect (s-trim (dom-texts (dom-by-class d "team-name__long")))))
(setq aways (cl-loop for d in aways-dom
collect (s-trim (dom-texts (dom-by-class d "team-name__long")))))
(when (string-equal data-type "results")
(setq results (cl-loop for h-d in homes-dom
for a-d in aways-dom
collect (concat (dom-texts (dom-by-class h-d "football-team__score"))
"-"
(dom-texts (dom-by-class a-d "football-team__score")))))
(setq dates (let* ((dates-dom (dom-by-class dom "football-matches__day")))
(-flatten (-concat (cl-loop for date-dom in dates-dom
collect (let* ((results-dom (dom-by-class date-dom "football-match__team football-match__team--home football-team"))
(day (dom-strings (dom-by-class dates-dom "date-divider"))))
(cl-loop for d in results-dom
collect day))))))))
(setq league-data `(("date" . ,dates)
("time" . ,times)
("source-time-utc-offset" . ,utc-offsets)
("home" . ,homes)
("away" . ,aways)
("result" . ,results)))
(if club
(soccer--filter-league-data-alist-by-club club league-data)
league-data)))))
(defun soccer--filter-league-data-alist-by-club (club league-data-alist)
"Filter the LEAGUE-DATA-ALIST by CLUB."
(let* ((indices1 (-elem-indices club (cdr (assoc "home" league-data-alist))))
(indices2 (-elem-indices club (cdr (assoc "away" league-data-alist))))
(indices (-sort '< (-concat indices1 indices2)))
(items (mapcar 'car league-data-alist)))
(cl-loop for item in items
collect (cons item (cl-loop for index in indices
collect (nth index (cdr (assoc item league-data-alist))))))))
(defun soccer--get-league-data-fixture-strings (dates times homes aways source-time-utc-offsets n)
"Get the fixtures strings to show in buffer for given DATES, TIMES, HOMES, AWAYS, SOURCE-TIME-UTC-OFFSETS and N, where n is the nth in the results."
(let* ((date (nth n dates))
(time (nth n times))
(source-time-utc-offset (nth n source-time-utc-offsets))
(local-time-list (soccer-time--get-local-time-list time date "-" 2 1 0 source-time-utc-offset soccer-time-local-time-utc-offset))
(local-min (nth 0 local-time-list))
(local-hour (nth 1 local-time-list))
(local-A/P (nth 2 local-time-list))
(match-day-local (nth 3 local-time-list))
(match-day-num-local (nth 4 local-time-list))
(match-month-local (nth 5 local-time-list))
(match-year-local (nth 6 local-time-list))
(home (nth n homes))
(away (nth n aways))
(time-till-kickoff-list (soccer-time--get-time-till-kick-off time date "-" 2 1 0 source-time-utc-offset soccer-time-local-time-utc-offset))
(days-remain (nth 0 time-till-kickoff-list))
(hours-remain (nth 1 time-till-kickoff-list))
(mins-remain (nth 2 time-till-kickoff-list))
time-till-kickoff-string
local-time-string)
(setq local-time-string (format "UTC%s %s %s %s %s %s:%s %s" soccer-time-local-time-utc-offset match-year-local match-month-local (format "%02d" match-day-num-local) match-day-local (format "%02d" local-hour) (format "%02d" local-min) local-A/P))
(setq time-till-kickoff-string (format "⟶ %s %s %s"
(if (> days-remain 0) (format "%2s days" days-remain) "")
(if (> hours-remain 0) (format "%2s hours" hours-remain) "")
(cond ((>= mins-remain 0) (format "%2s mins until kickoff" mins-remain))
((and (< mins-remain 0) (> mins-remain (- 95))) "match is live now.")
(t "match has finished"))))
(format "%s %40s - %-40s %s" (propertize local-time-string 'face 'soccer-face-local-time) (propertize home 'face 'soccer-face-fixtures) (propertize away 'face 'soccer-face-fixtures) (propertize time-till-kickoff-string 'face 'soccer-face-time-to-kickoff))))
(defun soccer--get-league-data-results-strings (dates homes aways results n)
"Get the fixtures stings to show in buffer for given DATES, HOMES, AWAYS, RESULTS and N, where is the nth in the results."
(let* ((date (nth n dates))
(home (nth n homes))
(away (nth n aways))
(result (nth n results))
(home-goals (substring result 0 1))
(away-goals (substring result 2)))
(format "%s %s" date
(cond ((> (string-to-number home-goals) (string-to-number away-goals))
(format "%s - %s" (propertize (concat home " " home-goals) 'face 'soccer-face-win) (propertize (concat away-goals " " away) 'face 'soccer-face-loss)))
((< (string-to-number home-goals) (string-to-number away-goals))
(format "%s - %s" (propertize (concat home " " home-goals) 'face 'soccer-face-loss) (propertize (concat away-goals " " away) 'face 'soccer-face-win)))
((= (string-to-number home-goals) (string-to-number away-goals))
(format "%s - %s" (propertize (concat home " " home-goals) 'face 'soccer-face-draw) (propertize (concat away-goals " " away) 'face 'soccer-face-draw)))))))
(defun soccer--get-league-data (league data-type num-of-results &optional club)
"Get NUM-OF-RESULT number of DATA-TYPE for a LEAGUE. Optionally filter by CLUB."
(let* ((league-data (soccer--get-league-data-alist league data-type club))
(dates (cdr (assoc "date" league-data)))
(times (cdr (assoc "time" league-data)))
(homes (cdr (assoc "home" league-data)))
(aways (cdr (assoc "away" league-data)))
(source-time-utc-offsets (cdr (assoc "source-time-utc-offset" league-data)))
results
msg-str
(number-of-results (length dates)))
(when (string-equal data-type "results")
(setq results (cdr (assoc "result" league-data))))
(if num-of-results
(setq num-of-results (min number-of-results num-of-results))
(setq num-of-results number-of-results))
(setq msg-str (cl-loop for n from 0 to (1- num-of-results)
collect (cond ((string-equal data-type "fixtures") (soccer--get-league-data-fixture-strings dates times homes aways source-time-utc-offsets n))
((string-equal data-type "results") (soccer--get-league-data-results-strings dates homes aways results n)))))
(message "%s" (string-join msg-str "\n"))))
(defun soccer--get-league-data-in-org (league data-type num-of-results &optional club)
"Get NUM-OF-RESULT number of DATA-TYPE for a CLUB (optional) of a LEAGUE."
(let* ((league-data (soccer--get-league-data-alist league data-type club))
(dates (cdr (assoc "date" league-data)))
(times (cdr (assoc "time" league-data)))
(homes (cdr (assoc "home" league-data)))
(aways (cdr (assoc "away" league-data)))
results
(file-name (format "%ssoccer_%s_%s.org" (temporary-file-directory) club data-type))
(number-of-results (length dates)))
(when (string-equal data-type "results")
(setq results (cdr (assoc "result" league-data))))
(if num-of-results
(setq num-of-results (min number-of-results num-of-results))
(setq num-of-results number-of-results))
(with-temp-buffer
(insert (format "* %s for %s\n\n" data-type club))
(insert "|--|--|--|--|--|\n")
(insert (format "|Date |Time | Home | Away | %s \n" (if (string-equal data-type "results") "score |" "")))
(insert "|--|--|--|--|\n")
(dolist (n (number-sequence 0 (1- num-of-results)))
(let* ((date (nth n dates))
(time (if times (nth n times) ""))
(home (nth n homes))
(away (nth n aways))
result)
(when (string-equal data-type "results")
(setq result (nth n results)))
(insert (format "|%s|%s|%s|%s|%s\n" date time home away (if (string-equal data-type "results") (format "%s|" result) "")))
(insert "|--|--|--|--|\n")))
(forward-line (* -1 num-of-results))
(org-table-align)
(write-region (point-min) (point-max) file-name)
(find-file file-name))))
(defun soccer--get-league-table-data-alist (league)
"Get data alist for standing table for a LEAGUE."
(let* ((url (concat (soccer--get-league-url league) "/table")))
(with-current-buffer (url-retrieve-synchronously url)
(let* ((dom (libxml-parse-html-region (point-min) (point-max)))
(dom-strings-list (-remove #'string-blank-p (dom-strings (nth 0 (dom-by-tag dom 'table)))))
(results-strings-list (-partition 15 (-drop 12 dom-strings-list)))
teams
played
wins
losses
goal-for
goal-against
current-form-list)
(setq teams-full-name (cl-loop for result in results-strings-list
collect (string-trim (nth 1 result))))
(setq teams (cl-loop for result in results-strings-list
collect ""))
(setq played (cl-loop for result in results-strings-list
collect (nth 2 result)))
(setq wins (cl-loop for result in results-strings-list
collect (nth 3 result)))
(setq losses (cl-loop for result in results-strings-list
collect (nth 5 result)))
(setq goal-for (cl-loop for result in results-strings-list
collect (nth 6 result)))
(setq goal-against (cl-loop for result in results-strings-list
collect (nth 7 result)))
(setq current-form-list (cl-loop for result in results-strings-list
collect (-take-last 5 result)))
`(("teams-full-name" . ,teams-full-name)
("teams" . ,teams)
("played" . ,played)
("wins" . ,wins)
("losses" . ,losses)
("goal-for" . ,goal-for)
("goal-against" . ,goal-against)
("current-form-list" . ,current-form-list))))))
(defun soccer--table-data (league num top/bottom)
"Get table for LEAGUE with TOP/BOTTOM NUM teams."
(let* ((data-alist (soccer--get-league-table-data-alist league))
(team-names (cdr (assoc "teams-full-name" data-alist)))
(teams (cdr (assoc "teams" data-alist)))
(played (cdr (assoc "played" data-alist)))
(wins (cdr (assoc "wins" data-alist)))
(losses (cdr (assoc "losses" data-alist)))
(goal-for (cdr (assoc "goal-for" data-alist)))
(goal-against (cdr (assoc "goal-against" data-alist)))
(current-form-list (cdr (assoc "current-form-list" data-alist)))
(num-result (min num (length teams)))
(start (if (string-equal top/bottom "top") 0 (- (length teams) num-result)))
(stop (if (string-equal top/bottom "top") (1- num-result) (1- (length teams))))
(header-string (format "Rank Team %-20s Played Won Lost Drawn GF GA GD Points Form" "Name"))
table-string)
(setq table-string (cl-loop for n from start to stop
collect (let ((team (nth n teams))
(team-name (nth n team-names))
(matches (nth n played))
(win (string-to-number (nth n wins)))
(loss (string-to-number (nth n losses)))
(goalF (string-to-number (nth n goal-for)))
(goalA (string-to-number (nth n goal-against)))
(current-form (nth n current-form-list))
current-form-string-list
draw)
(setq draw (- (string-to-number matches) win loss))
(setq current-form-string-list (cl-loop for form-string-full in current-form
collect (let ((form-string (upcase (substring form-string-full 0 1))))
(cond ((string-equal form-string "W") (propertize form-string 'face 'soccer-face-win))
((string-equal form-string "L") (propertize form-string 'face 'soccer-face-loss))
((string-equal form-string "D") (propertize form-string 'face 'soccer-face-draw))))))
(format "%3s %4s %-20s %5s %4s %4s %5s %4s %3s %3s %5s %8s" (1+ n) team team-name matches (propertize (format "%s" win) 'face 'soccer-face-win) (propertize (format "%s" loss) 'face 'soccer-face-loss) (propertize (format "%s" draw) 'face 'soccer-face-draw) goalF goalA (- goalF goalA) (+ (* 3 win) draw) (string-join current-form-string-list)))))
(string-join (-insert-at 0 header-string table-string) "\n")))
;;;###autoload
(defun soccer-table-top-4 (league)
"Get table for LEAGUE with top 4 teams."
(interactive
(list (completing-read "league: " (soccer--get-league-names))))
(message (soccer--table-data league 4 "top")))
;;;###autoload
(defun soccer-table-bottom-4 (league)
"Get table for LEAGUE with bottom 4 teams."
(interactive
(list (completing-read "league: " (soccer--get-league-names))))
(message (soccer--table-data league 4 "bottom")))
;;;###autoload
(defun soccer-table (league)
"Get full rank table of a LEAGUE."
(interactive
(list (completing-read "league: " (soccer--get-league-names))))
(let ((buffer-name (concat "*soccer-rank-table-" league "*")))
(generate-new-buffer buffer-name)
(with-current-buffer buffer-name
(insert (soccer--table-data league 30 "top")))
(switch-to-buffer-other-window buffer-name)))
;;;###autoload
(defun soccer-fixtures-full-in-org (league club)
"Full fixtures of CLUB of LEAGUE saved in a org file."
(interactive
(let* ((league-name (completing-read "league: " (soccer--get-league-names)))
(club-name (completing-read "club: " (mapcar 'car (soccer-leagues--get-club-names-and-urls league-name)))))
(list league-name club-name)))
(soccer--get-league-data-in-org league "fixtures" nil club))
;;;###autoload
(defun soccer-fixtures-next (league club)
"The next match of CLUB of LEAGUE."
(interactive
(let* ((league-name (completing-read "league: " (soccer--get-league-names)))
(club-name (completing-read "club: " (mapcar 'car (soccer-leagues--get-club-names-and-urls league-name)))))
(list league-name club-name)))
(soccer--get-league-data league "fixtures" 1 club))
;;;###autoload
(defun soccer-fixtures-all-clubs (league)
"The next matches in a LEAGUE."
(interactive
(list (completing-read "league: " (soccer--get-league-names))))
(soccer--get-league-data league "fixtures" 10))
;;;###autoload
(defun soccer-fixtures-next-5 (league club)
"The next 5 matches in fixtures of CLUB of LEAGUE."
(interactive
(let* ((league-name (completing-read "league: " (soccer--get-league-names)))
(club-name (completing-read "club: " (mapcar 'car (soccer-leagues--get-club-names-and-urls league-name)))))
(list league-name club-name)))
(soccer--get-league-data league "fixtures" 5 club))
;;;###autoload
(defun soccer-results-full-in-org (league club)
"Full results of CLUB of LEAGUE saved in a org file."
(interactive
(let* ((league-name (completing-read "league: " (soccer--league-names)))
(club-name (completing-read "club: " (mapcar 'car (soccer-leagues--get-club-names-and-urls league-name)))))
(list league-name club-name)))
(soccer--get-league-data-in-org league club "results" nil))
;;;###autoload
(defun soccer-results-last (league club)
"The last result of CLUB of LEAGUE."
(interactive
(let* ((league-name (completing-read "league: " (soccer--get-league-names)))
(club-name (completing-read "club: " (mapcar 'car (soccer-leagues--get-club-names-and-urls league-name)))))
(list league-name club-name)))
(soccer--get-league-data league "results" 1 club))
;;;###autoload
(defun soccer-results-last-5 (league club)
"The last 5 results of CLUB of LEAGUE."
(interactive
(let* ((league-name (completing-read "league: " (soccer--get-league-names)))
(club-name (completing-read "club: " (mapcar 'car (soccer-leagues--get-club-names-and-urls league-name)))))
(list league-name club-name)))
(soccer--get-league-data league "results" 5 club))
;;;###autoload
(defun soccer-results-all-clubs (league)
"All the latest results in a LEAGUE."
(interactive
(list (completing-read "league: " (soccer--get-league-names))))
(soccer--get-league-data league "results" 10))
;; score card
(defun soccer--get-scorecard-alist (date home away)
"Get the live scorecard for a match between HOME and AWAY on DATE. DATE should in the format DD-MM-YYYY."
(let* ((live-match-url (downcase (replace-regexp-in-string " " "-" (concat "https://www.scorespro.com/soccer/livescore/" home "-vs-" away "/" date "/")))))
(with-current-buffer (url-retrieve-synchronously live-match-url)
(let* ((dom (libxml-parse-html-region (point-min) (point-max)))
(title (dom-strings (dom-by-tag dom 'title)))
(home-goal-scorers (dom-strings (dom-by-class dom "sp-goalscorers__home-team")))
(away-goal-scorers (dom-strings (dom-by-class dom "sp-goalscorers__away-team")))
(home-goals (/ (length home-goal-scorers) 2))
(away-goals (/ (length away-goal-scorers) 2))
(home-scorecard-strings-list (delete-dups (-partition 2 home-goal-scorers)))
(away-scorecard-strings-list (delete-dups (-partition 2 away-goal-scorers))))
(when title
`(("title" . ,title)
("home" . ,home)
("away" . ,away)
("home-goals" . ,home-goals)
("away-goals" . ,away-goals)
("home-scorecard-list" . ,home-scorecard-strings-list)
("away-scorecard-list" . ,away-scorecard-strings-list)))))))
(defun soccer--all-clubs ()
"Get all club names."
(delete-dups (-flatten (cl-loop for leagues in (soccer--league-names)
collect (mapcar 'car (cdr (assoc leagues soccer-leagues--leagues-alist)))))))
(defun soccer-scorecard (date home away)
"Get the socrecard for a match between HOME and AWAY on a DATE. Enter DATE in YYYY-MM-DD format if entering it manually. If the input is from `org-read-date' calendar popup then it is in YYYY-MM-DD format by default."
(interactive
(list (org-read-date nil nil nil "Date of the match: ") (completing-read "home/away club/nation: " (soccer--all-clubs)) (completing-read "away/home club/nation: " (soccer--all-clubs))))
(let* ((day-moth-year (soccer-time--get-day-month-year-number-from-date date "-" 2 1 0))
(dd (-first-item day-moth-year))
(mm (-second-item day-moth-year))
(yyyy (-third-item day-moth-year))
(new-date (format "%02d-%02d-%04d" dd mm yyyy))
(scorecard-alist (or (soccer--get-scorecard-alist new-date home away) (soccer--get-scorecard-alist new-date away home)))
(title (cdr (assoc "title" scorecard-alist))))
(if title
(let* ((home-goals (cdr (assoc "home-goals" scorecard-alist)))
(away-goals (cdr (assoc "away-goals" scorecard-alist)))
(home-team (cdr (assoc "home" scorecard-alist)))
(away-team (cdr (assoc "away" scorecard-alist)))
(home-scorecard-list (cdr (assoc "home-scorecard-list" scorecard-alist)))
(away-scorecard-list (cdr (assoc "away-scorecard-list" scorecard-alist)))
(home-scorecard-goals (delete-dups (cl-loop for g in home-scorecard-list
collect (format "%s(%s)" (-second-item g) (-first-item g)))))
(away-scorecard-goals (delete-dups (cl-loop for g in away-scorecard-list
collect (format "%s(%s)" (-first-item g) (-second-item g)))))
(home-scorecard (format "%-20s %s: %s" home-team (length home-scorecard-goals) (string-join home-scorecard-goals ", ")))
(away-scorecard (format "%-20s %s: %s" away-team (length away-scorecard-goals) (string-join away-scorecard-goals ", "))))
(message (format "%s\n%s\n%s" (propertize (-first-item title) 'face 'soccer-face-scorecard-header)
(propertize home-scorecard 'face (cond ((> home-goals away-goals) 'soccer-face-win)
((= home-goals away-goals) 'soccer-face-draw)
((< home-goals away-goals) 'soccer-face-loss)))
(propertize away-scorecard 'face (cond ((> home-goals away-goals) 'soccer-face-loss)
((= home-goals away-goals) 'soccer-face-draw)
((< home-goals away-goals) 'soccer-face-win))))))
(message "Result not found. Probably entered wrong date/name?\nTry `soccer-fixtures-next` to get the correct match info."))))
;;; Add soccer schedules for favourite matches
(defcustom soccer-schedule-league-team-alist '()
"An alist where each element is (league team).
This is used to add the schedules for the teams to agenda."
:type '(alist :value-type (group integer))
:group 'soccer)
(defun soccer--compute-delta-weeks (time)
"Compute the number of weeks between now and TIME.
TIME should be in org-time-stamp format."
(let* ((time-now (current-time))
(time-given (org-time-string-to-time time))
(time-delta (time-subtract time-given time-now)))
(/ (time-to-number-of-days time-delta) 7)))
(defun soccer--get-local-timestamp (time date source-time-utc-offset)
"Get local time stamp using origin DATE and TIME and SOURCE-TIME-UTC-OFFSET."
(let* ((delta-offset-seconds (* 60 (- (timezone-zone-to-minute soccer-time-local-time-utc-offset) (timezone-zone-to-minute source-time-utc-offset))))
(time-origin (org-time-string-to-time (format "%s %s" date time)))
(time-local (time-add time-origin (seconds-to-time delta-offset-seconds))))
(format-time-string "%F %R" time-local)))
(defun soccer--get-schedule-data (league team num-weeks)
"Create the schedule for TEAM in a LEAGUE in the next NUM-WEEKS."
(let* ((data (soccer--get-league-data-alist league "fixtures" team))
(dates (cdr (assoc "date" data)))
(time (cdr (assoc "time" data)))
(home (cdr (assoc "home" data)))
(away (cdr (assoc "away" data)))
(utc-offset (cdr (assoc "source-time-utc-offset" data))))
(cl-loop for d in data
for date in dates
for ti in time
for h in home
for a in away
for offset in utc-offset
if (< (soccer--compute-delta-weeks (soccer--get-local-timestamp ti date offset)) num-weeks)
collect (list (format "%s: %s vs %s" league h a) (format "SCHEDULED: <%s>" (soccer--get-local-timestamp ti date offset))))))
(defcustom soccer-schedule-dir (expand-file-name "~/Dropbox/org/")
"Directory to store soccer schedules.
Remember to add this in the list of agenda files if it's not already added."
:type 'string
:group 'soccer)
(defcustom soccer-schedule-keyword "FOOTBALL"
"Keyword to represent fixture schedule in agenda view."
:type 'string
:group 'soccer)
(defun soccer--get-schedule-file-name (league)
"Get the file name based on LEAGUE."
(file-name-concat soccer-schedule-dir (format "%s.org" (string-replace " " "-" league))))
(defun soccer--create-initial-file (league)
"Create a file for LEAGUE to store schedules."
(let ((file-name (soccer--get-schedule-file-name league)))
(with-temp-buffer
(insert (format "#+TITLE: Schedules for %s\n\n" league))
(append-to-file (point-min) (point-max) file-name))))
(defun soccer--schedule-existsp (header file-name)
"Check if a schedule HEADER aleady exists inside a file of FILE-NAME."
(let ((content (with-temp-buffer
(insert-file-contents file-name)
(forward-line)
(buffer-substring-no-properties (point) (point-max)))))
(string-match-p header content)))
;;;###autoload
(defun soccer-schedule (league team num-weeks)
"Add the schedules of TEAM in LEAGUE for the next NUM-WEEKS."
(interactive
(let* ((league-name (completing-read "league: " (soccer--get-league-names)))
(team-name (completing-read "team: " (mapcar 'car (soccer-leagues--get-club-names-and-urls league-name))))
(num-weeks (completing-read "for number of weeks: " (mapcar #'number-to-string (number-sequence 1 38)))))
(list league-name team-name (string-to-number num-weeks))))
(let* ((file-name (soccer--get-schedule-file-name league))
(data (soccer--get-schedule-data league team num-weeks))
(headers (mapcar #'car data))
(timestamps (mapcar #'(lambda (d) (car (cdr d))) data)))
(unless (file-exists-p file-name)
(soccer--create-initial-file league))
(with-temp-buffer
(cl-loop for h in headers
for ts in timestamps
when (not (soccer--schedule-existsp h file-name))
do (insert (format "* %s %s\n%s\n\n" soccer-schedule-keyword h ts)))
(append-to-file (point-min) (point-max) file-name))
(with-current-buffer (find-file-noselect file-name)
(write-file file-name)
(kill-buffer))))
;;;###autoload
(defun soccer-schedule-league (league num-weeks)
"Schedule fixtures of all clubs for a LEAGUE for next NUM-WEEKS."
(interactive
(let* ((league (completing-read "league: " (soccer--get-league-names)))
(num-weeks (completing-read "for number of weeks: " (mapcar #'number-to-string (number-sequence 1 38)))))
(list league (string-to-number num-weeks))))
(let* ((team-names (mapcar 'car (soccer-leagues--get-club-names-and-urls league))))
(dolist (team team-names)
(soccer-schedule league team num-weeks))))
;;;###autoload
(defun soccer-schedule-remove-past-fixtures (league)
"Remove past fixtures of a LEAGUE."
(interactive
(let* ((league-name (completing-read "league: " (soccer--get-league-names))))
(list league-name)))
(let* ((file-name (soccer--get-schedule-file-name league)))
;;; check if any file for the league exists
(unless (file-exists-p file-name)
(user-error "No schedule file for %s exists!" league))
(with-current-buffer (find-file-noselect file-name)
(goto-char (point-min))
;;; forward line and look for the timestamps. For a given timestamp, check if it
;;; is in the past and if yes then remove that fixture and go on till the end of
;;; the file.
(while (not (eobp))
(let* ((current-line (thing-at-point 'line))
(timestamp-beg (string-match "<" current-line)))
(if timestamp-beg
;;; get the timestamp and then see if it is in the past
(let* ((timestamp-end (string-match ">" current-line))
(timestamp (substring current-line (1+ timestamp-beg) timestamp-end)))
;;; check if it is in the past
(when (< (time-to-seconds (time-subtract (org-time-string-to-time timestamp) (current-time))) 0)
;;; remove the schedule
;;; go to the header
(forward-line -1)
(let* ((beg (line-beginning-position)))
(forward-line 2)
;;; delete the line break also unless it is the end of buffer
;;; by adding 1 to end position
(delete-region beg (+ (line-end-position) (if (eobp) 0 1))))))))
(forward-line))
(write-file file-name))))
;;;###autoload
(defun soccer-schedule-remove-league (league)
"Remove all fixtures of a LEAGUE.
This will remove the org file for the given league."
(interactive
(let* ((league-name (completing-read "league: " (soccer--get-league-names))))
(list league-name)))
(let* ((file-name (soccer--get-schedule-file-name league)))
;;; check if any file for the league exists
(unless (file-exists-p file-name)
(user-error "No schedule file for %s exists!" league))
(delete-file file-name)))
(provide 'soccer)
;;; soccer.el ends here