Mercurial > libervia-web
annotate src/browser/sat_browser/radiocol.py @ 649:ccf95ec87005 frontends_multi_profiles
browser_side: fixes issue when placing a LiberviaWidget and some cells of the WidgetsPanel are empty
author | souliane <souliane@mailoo.org> |
---|---|
date | Fri, 27 Feb 2015 00:55:07 +0100 |
parents | bbdc5357dc00 |
children |
rev | line source |
---|---|
127 | 1 #!/usr/bin/python |
2 # -*- coding: utf-8 -*- | |
3 | |
339
2067d6241927
fixed docstrings wrong usage for licence informations
Goffi <goffi@goffi.org>
parents:
301
diff
changeset
|
4 # Libervia: a Salut à Toi frontend |
340 | 5 # Copyright (C) 2011, 2012, 2013, 2014 Jérôme Poisson <goffi@goffi.org> |
127 | 6 |
339
2067d6241927
fixed docstrings wrong usage for licence informations
Goffi <goffi@goffi.org>
parents:
301
diff
changeset
|
7 # This program is free software: you can redistribute it and/or modify |
2067d6241927
fixed docstrings wrong usage for licence informations
Goffi <goffi@goffi.org>
parents:
301
diff
changeset
|
8 # it under the terms of the GNU Affero General Public License as published by |
2067d6241927
fixed docstrings wrong usage for licence informations
Goffi <goffi@goffi.org>
parents:
301
diff
changeset
|
9 # the Free Software Foundation, either version 3 of the License, or |
2067d6241927
fixed docstrings wrong usage for licence informations
Goffi <goffi@goffi.org>
parents:
301
diff
changeset
|
10 # (at your option) any later version. |
127 | 11 |
339
2067d6241927
fixed docstrings wrong usage for licence informations
Goffi <goffi@goffi.org>
parents:
301
diff
changeset
|
12 # This program is distributed in the hope that it will be useful, |
2067d6241927
fixed docstrings wrong usage for licence informations
Goffi <goffi@goffi.org>
parents:
301
diff
changeset
|
13 # but WITHOUT ANY WARRANTY; without even the implied warranty of |
2067d6241927
fixed docstrings wrong usage for licence informations
Goffi <goffi@goffi.org>
parents:
301
diff
changeset
|
14 # MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the |
2067d6241927
fixed docstrings wrong usage for licence informations
Goffi <goffi@goffi.org>
parents:
301
diff
changeset
|
15 # GNU Affero General Public License for more details. |
127 | 16 |
339
2067d6241927
fixed docstrings wrong usage for licence informations
Goffi <goffi@goffi.org>
parents:
301
diff
changeset
|
17 # You should have received a copy of the GNU Affero General Public License |
2067d6241927
fixed docstrings wrong usage for licence informations
Goffi <goffi@goffi.org>
parents:
301
diff
changeset
|
18 # along with this program. If not, see <http://www.gnu.org/licenses/>. |
127 | 19 |
243
63e9b680d3e7
browser_side, blog: better PEP8 compliance
souliane <souliane@mailoo.org>
parents:
165
diff
changeset
|
20 import pyjd # this is dummy in pyjs |
439
d52f529a6d42
browser side: use of new log system (first draft):
Goffi <goffi@goffi.org>
parents:
422
diff
changeset
|
21 from sat.core.log import getLogger |
d52f529a6d42
browser side: use of new log system (first draft):
Goffi <goffi@goffi.org>
parents:
422
diff
changeset
|
22 log = getLogger(__name__) |
449
981ed669d3b3
/!\ reorganize all the file hierarchy, move the code and launching script to src:
souliane <souliane@mailoo.org>
parents:
439
diff
changeset
|
23 from sat_frontends.tools.misc import DEFAULT_MUC |
981ed669d3b3
/!\ reorganize all the file hierarchy, move the code and launching script to src:
souliane <souliane@mailoo.org>
parents:
439
diff
changeset
|
24 from sat.core.i18n import _ |
481
bbdc5357dc00
browser and server sides: refactor HTTP request result values + handle "NoReply" error
souliane <souliane@mailoo.org>
parents:
467
diff
changeset
|
25 from constants import Const as C |
449
981ed669d3b3
/!\ reorganize all the file hierarchy, move the code and launching script to src:
souliane <souliane@mailoo.org>
parents:
439
diff
changeset
|
26 |
127 | 27 from pyjamas.ui.VerticalPanel import VerticalPanel |
28 from pyjamas.ui.HorizontalPanel import HorizontalPanel | |
129
dd0d39ae7d24
RadioCol: song preloading + fonctionnal players
Goffi <goffi@goffi.org>
parents:
128
diff
changeset
|
29 from pyjamas.ui.FlexTable import FlexTable |
127 | 30 from pyjamas.ui.FormPanel import FormPanel |
31 from pyjamas.ui.Label import Label | |
32 from pyjamas.ui.Button import Button | |
33 from pyjamas.ui.ClickListener import ClickHandler | |
128 | 34 from pyjamas.ui.Hidden import Hidden |
286
e76ec07be8e5
browser_side (plugin radiocol): the UI uses CaptionPanel
souliane <souliane@mailoo.org>
parents:
285
diff
changeset
|
35 from pyjamas.ui.CaptionPanel import CaptionPanel |
341
31c105017d6b
browser_side: radiocol current song playback will jump to time when a user joins a running session
souliane <souliane@mailoo.org>
parents:
340
diff
changeset
|
36 from pyjamas.media.Audio import Audio |
127 | 37 from pyjamas import Window |
129
dd0d39ae7d24
RadioCol: song preloading + fonctionnal players
Goffi <goffi@goffi.org>
parents:
128
diff
changeset
|
38 from pyjamas.Timer import Timer |
127 | 39 |
467 | 40 import html_tools |
41 import file_tools | |
127 | 42 |
43 | |
129
dd0d39ae7d24
RadioCol: song preloading + fonctionnal players
Goffi <goffi@goffi.org>
parents:
128
diff
changeset
|
44 class MetadataPanel(FlexTable): |
127 | 45 |
46 def __init__(self): | |
129
dd0d39ae7d24
RadioCol: song preloading + fonctionnal players
Goffi <goffi@goffi.org>
parents:
128
diff
changeset
|
47 FlexTable.__init__(self) |
243
63e9b680d3e7
browser_side, blog: better PEP8 compliance
souliane <souliane@mailoo.org>
parents:
165
diff
changeset
|
48 title_lbl = Label("title:") |
129
dd0d39ae7d24
RadioCol: song preloading + fonctionnal players
Goffi <goffi@goffi.org>
parents:
128
diff
changeset
|
49 title_lbl.setStyleName('radiocol_metadata_lbl') |
dd0d39ae7d24
RadioCol: song preloading + fonctionnal players
Goffi <goffi@goffi.org>
parents:
128
diff
changeset
|
50 artist_lbl = Label("artist:") |
dd0d39ae7d24
RadioCol: song preloading + fonctionnal players
Goffi <goffi@goffi.org>
parents:
128
diff
changeset
|
51 artist_lbl.setStyleName('radiocol_metadata_lbl') |
dd0d39ae7d24
RadioCol: song preloading + fonctionnal players
Goffi <goffi@goffi.org>
parents:
128
diff
changeset
|
52 album_lbl = Label("album:") |
dd0d39ae7d24
RadioCol: song preloading + fonctionnal players
Goffi <goffi@goffi.org>
parents:
128
diff
changeset
|
53 album_lbl.setStyleName('radiocol_metadata_lbl') |
243
63e9b680d3e7
browser_side, blog: better PEP8 compliance
souliane <souliane@mailoo.org>
parents:
165
diff
changeset
|
54 self.title = Label("") |
129
dd0d39ae7d24
RadioCol: song preloading + fonctionnal players
Goffi <goffi@goffi.org>
parents:
128
diff
changeset
|
55 self.title.setStyleName('radiocol_metadata') |
dd0d39ae7d24
RadioCol: song preloading + fonctionnal players
Goffi <goffi@goffi.org>
parents:
128
diff
changeset
|
56 self.artist = Label("") |
dd0d39ae7d24
RadioCol: song preloading + fonctionnal players
Goffi <goffi@goffi.org>
parents:
128
diff
changeset
|
57 self.artist.setStyleName('radiocol_metadata') |
dd0d39ae7d24
RadioCol: song preloading + fonctionnal players
Goffi <goffi@goffi.org>
parents:
128
diff
changeset
|
58 self.album = Label("") |
dd0d39ae7d24
RadioCol: song preloading + fonctionnal players
Goffi <goffi@goffi.org>
parents:
128
diff
changeset
|
59 self.album.setStyleName('radiocol_metadata') |
243
63e9b680d3e7
browser_side, blog: better PEP8 compliance
souliane <souliane@mailoo.org>
parents:
165
diff
changeset
|
60 self.setWidget(0, 0, title_lbl) |
63e9b680d3e7
browser_side, blog: better PEP8 compliance
souliane <souliane@mailoo.org>
parents:
165
diff
changeset
|
61 self.setWidget(1, 0, artist_lbl) |
63e9b680d3e7
browser_side, blog: better PEP8 compliance
souliane <souliane@mailoo.org>
parents:
165
diff
changeset
|
62 self.setWidget(2, 0, album_lbl) |
63e9b680d3e7
browser_side, blog: better PEP8 compliance
souliane <souliane@mailoo.org>
parents:
165
diff
changeset
|
63 self.setWidget(0, 1, self.title) |
63e9b680d3e7
browser_side, blog: better PEP8 compliance
souliane <souliane@mailoo.org>
parents:
165
diff
changeset
|
64 self.setWidget(1, 1, self.artist) |
63e9b680d3e7
browser_side, blog: better PEP8 compliance
souliane <souliane@mailoo.org>
parents:
165
diff
changeset
|
65 self.setWidget(2, 1, self.album) |
129
dd0d39ae7d24
RadioCol: song preloading + fonctionnal players
Goffi <goffi@goffi.org>
parents:
128
diff
changeset
|
66 self.setStyleName("radiocol_metadata_pnl") |
dd0d39ae7d24
RadioCol: song preloading + fonctionnal players
Goffi <goffi@goffi.org>
parents:
128
diff
changeset
|
67 |
dd0d39ae7d24
RadioCol: song preloading + fonctionnal players
Goffi <goffi@goffi.org>
parents:
128
diff
changeset
|
68 def setTitle(self, title): |
dd0d39ae7d24
RadioCol: song preloading + fonctionnal players
Goffi <goffi@goffi.org>
parents:
128
diff
changeset
|
69 self.title.setText(title) |
dd0d39ae7d24
RadioCol: song preloading + fonctionnal players
Goffi <goffi@goffi.org>
parents:
128
diff
changeset
|
70 |
dd0d39ae7d24
RadioCol: song preloading + fonctionnal players
Goffi <goffi@goffi.org>
parents:
128
diff
changeset
|
71 def setArtist(self, artist): |
dd0d39ae7d24
RadioCol: song preloading + fonctionnal players
Goffi <goffi@goffi.org>
parents:
128
diff
changeset
|
72 self.artist.setText(artist) |
dd0d39ae7d24
RadioCol: song preloading + fonctionnal players
Goffi <goffi@goffi.org>
parents:
128
diff
changeset
|
73 |
dd0d39ae7d24
RadioCol: song preloading + fonctionnal players
Goffi <goffi@goffi.org>
parents:
128
diff
changeset
|
74 def setAlbum(self, album): |
dd0d39ae7d24
RadioCol: song preloading + fonctionnal players
Goffi <goffi@goffi.org>
parents:
128
diff
changeset
|
75 self.album.setText(album) |
127 | 76 |
243
63e9b680d3e7
browser_side, blog: better PEP8 compliance
souliane <souliane@mailoo.org>
parents:
165
diff
changeset
|
77 |
127 | 78 class ControlPanel(FormPanel): |
79 """Panel used to show controls to add a song, or vote for the current one""" | |
80 | |
285
4f0c2fea358a
browser_side (plugin radiocol): use the status label to give more information
souliane <souliane@mailoo.org>
parents:
284
diff
changeset
|
81 def __init__(self, parent): |
127 | 82 FormPanel.__init__(self) |
83 self.setEncoding(FormPanel.ENCODING_MULTIPART) | |
84 self.setMethod(FormPanel.METHOD_POST) | |
151
a159cc29b556
server side: file upload is now more generic:
Goffi <goffi@goffi.org>
parents:
131
diff
changeset
|
85 self.setAction("upload_radiocol") |
300
4f221f34bdc7
browser_side (plugins radiocol, xep-0054): handle upload errors
souliane <souliane@mailoo.org>
parents:
287
diff
changeset
|
86 self.timer_on = False |
285
4f0c2fea358a
browser_side (plugin radiocol): use the status label to give more information
souliane <souliane@mailoo.org>
parents:
284
diff
changeset
|
87 self._parent = parent |
286
e76ec07be8e5
browser_side (plugin radiocol): the UI uses CaptionPanel
souliane <souliane@mailoo.org>
parents:
285
diff
changeset
|
88 vPanel = VerticalPanel() |
127 | 89 |
283
0eba1c4f9c6f
browser_side (plugins radiocol, xep-0054): check for file size or type before uploading
souliane <souliane@mailoo.org>
parents:
275
diff
changeset
|
90 types = [('audio/ogg', '*.ogg', 'Ogg Vorbis Audio'), |
0eba1c4f9c6f
browser_side (plugins radiocol, xep-0054): check for file size or type before uploading
souliane <souliane@mailoo.org>
parents:
275
diff
changeset
|
91 ('video/ogg', '*.ogv', 'Ogg Vorbis Video'), |
394
ee61b0765d6c
browser_side: radiocol supports MP3
souliane <souliane@mailoo.org>
parents:
370
diff
changeset
|
92 ('application/ogg', '*.ogx', 'Ogg Vorbis Multiplex'), |
398
462fc3359ee3
browser_side: radiocol accepts mime type audio/mp3 in addition to audio/mpeg
souliane <souliane@mailoo.org>
parents:
397
diff
changeset
|
93 ('audio/mpeg', '*.mp3', 'MPEG-Layer 3'), |
462fc3359ee3
browser_side: radiocol accepts mime type audio/mp3 in addition to audio/mpeg
souliane <souliane@mailoo.org>
parents:
397
diff
changeset
|
94 ('audio/mp3', '*.mp3', 'MPEG-Layer 3'), |
462fc3359ee3
browser_side: radiocol accepts mime type audio/mp3 in addition to audio/mpeg
souliane <souliane@mailoo.org>
parents:
397
diff
changeset
|
95 ] |
467 | 96 self.file_upload = file_tools.FilterFileUpload("song", 10, types) |
286
e76ec07be8e5
browser_side (plugin radiocol): the UI uses CaptionPanel
souliane <souliane@mailoo.org>
parents:
285
diff
changeset
|
97 vPanel.add(self.file_upload) |
127 | 98 |
286
e76ec07be8e5
browser_side (plugin radiocol): the UI uses CaptionPanel
souliane <souliane@mailoo.org>
parents:
285
diff
changeset
|
99 hPanel = HorizontalPanel() |
130 | 100 self.upload_btn = Button("Upload song", getattr(self, "onBtnClick")) |
101 hPanel.add(self.upload_btn) | |
129
dd0d39ae7d24
RadioCol: song preloading + fonctionnal players
Goffi <goffi@goffi.org>
parents:
128
diff
changeset
|
102 self.status = Label() |
285
4f0c2fea358a
browser_side (plugin radiocol): use the status label to give more information
souliane <souliane@mailoo.org>
parents:
284
diff
changeset
|
103 self.updateStatus() |
130 | 104 hPanel.add(self.status) |
394
ee61b0765d6c
browser_side: radiocol supports MP3
souliane <souliane@mailoo.org>
parents:
370
diff
changeset
|
105 #We need to know the filename and the referee |
ee61b0765d6c
browser_side: radiocol supports MP3
souliane <souliane@mailoo.org>
parents:
370
diff
changeset
|
106 self.filename_field = Hidden('filename', '') |
ee61b0765d6c
browser_side: radiocol supports MP3
souliane <souliane@mailoo.org>
parents:
370
diff
changeset
|
107 hPanel.add(self.filename_field) |
285
4f0c2fea358a
browser_side (plugin radiocol): use the status label to give more information
souliane <souliane@mailoo.org>
parents:
284
diff
changeset
|
108 referee_field = Hidden('referee', self._parent.referee) |
394
ee61b0765d6c
browser_side: radiocol supports MP3
souliane <souliane@mailoo.org>
parents:
370
diff
changeset
|
109 hPanel.add(self.filename_field) |
285
4f0c2fea358a
browser_side (plugin radiocol): use the status label to give more information
souliane <souliane@mailoo.org>
parents:
284
diff
changeset
|
110 hPanel.add(referee_field) |
286
e76ec07be8e5
browser_side (plugin radiocol): the UI uses CaptionPanel
souliane <souliane@mailoo.org>
parents:
285
diff
changeset
|
111 vPanel.add(hPanel) |
127 | 112 |
286
e76ec07be8e5
browser_side (plugin radiocol): the UI uses CaptionPanel
souliane <souliane@mailoo.org>
parents:
285
diff
changeset
|
113 self.add(vPanel) |
127 | 114 self.addFormHandler(self) |
243
63e9b680d3e7
browser_side, blog: better PEP8 compliance
souliane <souliane@mailoo.org>
parents:
165
diff
changeset
|
115 |
285
4f0c2fea358a
browser_side (plugin radiocol): use the status label to give more information
souliane <souliane@mailoo.org>
parents:
284
diff
changeset
|
116 def updateStatus(self): |
300
4f221f34bdc7
browser_side (plugins radiocol, xep-0054): handle upload errors
souliane <souliane@mailoo.org>
parents:
287
diff
changeset
|
117 if self.timer_on: |
4f221f34bdc7
browser_side (plugins radiocol, xep-0054): handle upload errors
souliane <souliane@mailoo.org>
parents:
287
diff
changeset
|
118 return |
285
4f0c2fea358a
browser_side (plugin radiocol): use the status label to give more information
souliane <souliane@mailoo.org>
parents:
284
diff
changeset
|
119 # TODO: the status should be different if a song is being played or not |
4f0c2fea358a
browser_side (plugin radiocol): use the status label to give more information
souliane <souliane@mailoo.org>
parents:
284
diff
changeset
|
120 queue = self._parent.getQueueSize() |
4f0c2fea358a
browser_side (plugin radiocol): use the status label to give more information
souliane <souliane@mailoo.org>
parents:
284
diff
changeset
|
121 queue_data = self._parent.queue_data |
4f0c2fea358a
browser_side (plugin radiocol): use the status label to give more information
souliane <souliane@mailoo.org>
parents:
284
diff
changeset
|
122 if queue < queue_data[0]: |
4f0c2fea358a
browser_side (plugin radiocol): use the status label to give more information
souliane <souliane@mailoo.org>
parents:
284
diff
changeset
|
123 left = queue_data[0] - queue |
4f0c2fea358a
browser_side (plugin radiocol): use the status label to give more information
souliane <souliane@mailoo.org>
parents:
284
diff
changeset
|
124 self.status.setText("[we need %d more song%s]" % (left, "s" if left > 1 else "")) |
4f0c2fea358a
browser_side (plugin radiocol): use the status label to give more information
souliane <souliane@mailoo.org>
parents:
284
diff
changeset
|
125 elif queue < queue_data[1]: |
4f0c2fea358a
browser_side (plugin radiocol): use the status label to give more information
souliane <souliane@mailoo.org>
parents:
284
diff
changeset
|
126 left = queue_data[1] - queue |
4f0c2fea358a
browser_side (plugin radiocol): use the status label to give more information
souliane <souliane@mailoo.org>
parents:
284
diff
changeset
|
127 self.status.setText("[%d available spot%s]" % (left, "s" if left > 1 else "")) |
4f0c2fea358a
browser_side (plugin radiocol): use the status label to give more information
souliane <souliane@mailoo.org>
parents:
284
diff
changeset
|
128 elif queue >= queue_data[1]: |
4f0c2fea358a
browser_side (plugin radiocol): use the status label to give more information
souliane <souliane@mailoo.org>
parents:
284
diff
changeset
|
129 self.status.setText("[The queue is currently full]") |
4f0c2fea358a
browser_side (plugin radiocol): use the status label to give more information
souliane <souliane@mailoo.org>
parents:
284
diff
changeset
|
130 self.status.setStyleName('radiocol_status') |
127 | 131 |
132 def onBtnClick(self): | |
283
0eba1c4f9c6f
browser_side (plugins radiocol, xep-0054): check for file size or type before uploading
souliane <souliane@mailoo.org>
parents:
275
diff
changeset
|
133 if self.file_upload.check(): |
394
ee61b0765d6c
browser_side: radiocol supports MP3
souliane <souliane@mailoo.org>
parents:
370
diff
changeset
|
134 self.status.setText('[Submitting, please wait...]') |
ee61b0765d6c
browser_side: radiocol supports MP3
souliane <souliane@mailoo.org>
parents:
370
diff
changeset
|
135 self.filename_field.setValue(self.file_upload.getFilename()) |
ee61b0765d6c
browser_side: radiocol supports MP3
souliane <souliane@mailoo.org>
parents:
370
diff
changeset
|
136 if self.file_upload.getFilename().lower().endswith('.mp3'): |
ee61b0765d6c
browser_side: radiocol supports MP3
souliane <souliane@mailoo.org>
parents:
370
diff
changeset
|
137 self._parent._parent.host.showWarning('STATUS', 'For a better support, it is recommended to submit Ogg Vorbis file instead of MP3. You can convert your files easily, ask for help if needed!', 5000) |
283
0eba1c4f9c6f
browser_side (plugins radiocol, xep-0054): check for file size or type before uploading
souliane <souliane@mailoo.org>
parents:
275
diff
changeset
|
138 self.submit() |
0eba1c4f9c6f
browser_side (plugins radiocol, xep-0054): check for file size or type before uploading
souliane <souliane@mailoo.org>
parents:
275
diff
changeset
|
139 self.file_upload.setFilename("") |
243
63e9b680d3e7
browser_side, blog: better PEP8 compliance
souliane <souliane@mailoo.org>
parents:
165
diff
changeset
|
140 |
127 | 141 def onSubmit(self, event): |
142 pass | |
143 | |
130 | 144 def blockUpload(self): |
145 self.file_upload.setVisible(False) | |
146 self.upload_btn.setEnabled(False) | |
147 | |
148 def unblockUpload(self): | |
149 self.file_upload.setVisible(True) | |
150 self.upload_btn.setEnabled(True) | |
151 | |
285
4f0c2fea358a
browser_side (plugin radiocol): use the status label to give more information
souliane <souliane@mailoo.org>
parents:
284
diff
changeset
|
152 def setTemporaryStatus(self, text, style): |
4f0c2fea358a
browser_side (plugin radiocol): use the status label to give more information
souliane <souliane@mailoo.org>
parents:
284
diff
changeset
|
153 self.status.setText(text) |
4f0c2fea358a
browser_side (plugin radiocol): use the status label to give more information
souliane <souliane@mailoo.org>
parents:
284
diff
changeset
|
154 self.status.setStyleName('radiocol_upload_status_%s' % style) |
300
4f221f34bdc7
browser_side (plugins radiocol, xep-0054): handle upload errors
souliane <souliane@mailoo.org>
parents:
287
diff
changeset
|
155 self.timer_on = True |
4f221f34bdc7
browser_side (plugins radiocol, xep-0054): handle upload errors
souliane <souliane@mailoo.org>
parents:
287
diff
changeset
|
156 |
422
20c508f9b32a
browser side: fixed bad use of Timer
Goffi <goffi@goffi.org>
parents:
398
diff
changeset
|
157 def cb(timer): |
300
4f221f34bdc7
browser_side (plugins radiocol, xep-0054): handle upload errors
souliane <souliane@mailoo.org>
parents:
287
diff
changeset
|
158 self.timer_on = False |
4f221f34bdc7
browser_side (plugins radiocol, xep-0054): handle upload errors
souliane <souliane@mailoo.org>
parents:
287
diff
changeset
|
159 self.updateStatus() |
4f221f34bdc7
browser_side (plugins radiocol, xep-0054): handle upload errors
souliane <souliane@mailoo.org>
parents:
287
diff
changeset
|
160 |
4f221f34bdc7
browser_side (plugins radiocol, xep-0054): handle upload errors
souliane <souliane@mailoo.org>
parents:
287
diff
changeset
|
161 Timer(5000, cb) |
285
4f0c2fea358a
browser_side (plugin radiocol): use the status label to give more information
souliane <souliane@mailoo.org>
parents:
284
diff
changeset
|
162 |
127 | 163 def onSubmitComplete(self, event): |
164 result = event.getResults() | |
481
bbdc5357dc00
browser and server sides: refactor HTTP request result values + handle "NoReply" error
souliane <souliane@mailoo.org>
parents:
467
diff
changeset
|
165 if result == C.UPLOAD_OK: |
287
7a1dc69112b8
browser_side (plugin radiocol): send the current queue to new players
souliane <souliane@mailoo.org>
parents:
286
diff
changeset
|
166 # the song can still be rejected (not readable, full queue...) |
7a1dc69112b8
browser_side (plugin radiocol): send the current queue to new players
souliane <souliane@mailoo.org>
parents:
286
diff
changeset
|
167 self.setTemporaryStatus('[Your song has been submitted to the radio]', "ok") |
481
bbdc5357dc00
browser and server sides: refactor HTTP request result values + handle "NoReply" error
souliane <souliane@mailoo.org>
parents:
467
diff
changeset
|
168 elif result == C.UPLOAD_KO: |
285
4f0c2fea358a
browser_side (plugin radiocol): use the status label to give more information
souliane <souliane@mailoo.org>
parents:
284
diff
changeset
|
169 self.setTemporaryStatus('[Something went wrong during your song upload]', "ko") |
394
ee61b0765d6c
browser_side: radiocol supports MP3
souliane <souliane@mailoo.org>
parents:
370
diff
changeset
|
170 self._parent.radiocolSongRejected(_("The uploaded file has been rejected, only Ogg Vorbis and MP3 songs are accepted.")) |
300
4f221f34bdc7
browser_side (plugins radiocol, xep-0054): handle upload errors
souliane <souliane@mailoo.org>
parents:
287
diff
changeset
|
171 # TODO: would be great to re-use the original Exception class and message |
4f221f34bdc7
browser_side (plugins radiocol, xep-0054): handle upload errors
souliane <souliane@mailoo.org>
parents:
287
diff
changeset
|
172 # but it is lost in the middle of the traceback and encapsulated within |
4f221f34bdc7
browser_side (plugins radiocol, xep-0054): handle upload errors
souliane <souliane@mailoo.org>
parents:
287
diff
changeset
|
173 # a DBusException instance --> extract the data from the traceback? |
127 | 174 else: |
481
bbdc5357dc00
browser and server sides: refactor HTTP request result values + handle "NoReply" error
souliane <souliane@mailoo.org>
parents:
467
diff
changeset
|
175 Window.alert(_('Submit error: %s' % result)) |
283
0eba1c4f9c6f
browser_side (plugins radiocol, xep-0054): check for file size or type before uploading
souliane <souliane@mailoo.org>
parents:
275
diff
changeset
|
176 self.status.setText('') |
127 | 177 |
243
63e9b680d3e7
browser_side, blog: better PEP8 compliance
souliane <souliane@mailoo.org>
parents:
165
diff
changeset
|
178 |
341
31c105017d6b
browser_side: radiocol current song playback will jump to time when a user joins a running session
souliane <souliane@mailoo.org>
parents:
340
diff
changeset
|
179 class Player(Audio): |
129
dd0d39ae7d24
RadioCol: song preloading + fonctionnal players
Goffi <goffi@goffi.org>
parents:
128
diff
changeset
|
180 |
dd0d39ae7d24
RadioCol: song preloading + fonctionnal players
Goffi <goffi@goffi.org>
parents:
128
diff
changeset
|
181 def __init__(self, player_id, metadata_panel): |
341
31c105017d6b
browser_side: radiocol current song playback will jump to time when a user joins a running session
souliane <souliane@mailoo.org>
parents:
340
diff
changeset
|
182 Audio.__init__(self) |
129
dd0d39ae7d24
RadioCol: song preloading + fonctionnal players
Goffi <goffi@goffi.org>
parents:
128
diff
changeset
|
183 self._id = player_id |
dd0d39ae7d24
RadioCol: song preloading + fonctionnal players
Goffi <goffi@goffi.org>
parents:
128
diff
changeset
|
184 self.metadata = metadata_panel |
287
7a1dc69112b8
browser_side (plugin radiocol): send the current queue to new players
souliane <souliane@mailoo.org>
parents:
286
diff
changeset
|
185 self.timestamp = "" |
243
63e9b680d3e7
browser_side, blog: better PEP8 compliance
souliane <souliane@mailoo.org>
parents:
165
diff
changeset
|
186 self.title = "" |
63e9b680d3e7
browser_side, blog: better PEP8 compliance
souliane <souliane@mailoo.org>
parents:
165
diff
changeset
|
187 self.artist = "" |
63e9b680d3e7
browser_side, blog: better PEP8 compliance
souliane <souliane@mailoo.org>
parents:
165
diff
changeset
|
188 self.album = "" |
129
dd0d39ae7d24
RadioCol: song preloading + fonctionnal players
Goffi <goffi@goffi.org>
parents:
128
diff
changeset
|
189 self.filename = None |
341
31c105017d6b
browser_side: radiocol current song playback will jump to time when a user joins a running session
souliane <souliane@mailoo.org>
parents:
340
diff
changeset
|
190 self.played = False # True when the song is playing/has played, becomes False on preload |
31c105017d6b
browser_side: radiocol current song playback will jump to time when a user joins a running session
souliane <souliane@mailoo.org>
parents:
340
diff
changeset
|
191 self.setAutobuffer(True) |
31c105017d6b
browser_side: radiocol current song playback will jump to time when a user joins a running session
souliane <souliane@mailoo.org>
parents:
340
diff
changeset
|
192 self.setAutoplay(False) |
31c105017d6b
browser_side: radiocol current song playback will jump to time when a user joins a running session
souliane <souliane@mailoo.org>
parents:
340
diff
changeset
|
193 self.setVisible(False) |
31c105017d6b
browser_side: radiocol current song playback will jump to time when a user joins a running session
souliane <souliane@mailoo.org>
parents:
340
diff
changeset
|
194 |
287
7a1dc69112b8
browser_side (plugin radiocol): send the current queue to new players
souliane <souliane@mailoo.org>
parents:
286
diff
changeset
|
195 def preload(self, timestamp, filename, title, artist, album): |
129
dd0d39ae7d24
RadioCol: song preloading + fonctionnal players
Goffi <goffi@goffi.org>
parents:
128
diff
changeset
|
196 """preload the song but doesn't play it""" |
287
7a1dc69112b8
browser_side (plugin radiocol): send the current queue to new players
souliane <souliane@mailoo.org>
parents:
286
diff
changeset
|
197 self.timestamp = timestamp |
129
dd0d39ae7d24
RadioCol: song preloading + fonctionnal players
Goffi <goffi@goffi.org>
parents:
128
diff
changeset
|
198 self.filename = filename |
dd0d39ae7d24
RadioCol: song preloading + fonctionnal players
Goffi <goffi@goffi.org>
parents:
128
diff
changeset
|
199 self.title = title |
dd0d39ae7d24
RadioCol: song preloading + fonctionnal players
Goffi <goffi@goffi.org>
parents:
128
diff
changeset
|
200 self.artist = artist |
dd0d39ae7d24
RadioCol: song preloading + fonctionnal players
Goffi <goffi@goffi.org>
parents:
128
diff
changeset
|
201 self.album = album |
dd0d39ae7d24
RadioCol: song preloading + fonctionnal players
Goffi <goffi@goffi.org>
parents:
128
diff
changeset
|
202 self.played = False |
467 | 203 self.setSrc("radiocol/%s" % html_tools.html_sanitize(filename)) |
439
d52f529a6d42
browser side: use of new log system (first draft):
Goffi <goffi@goffi.org>
parents:
422
diff
changeset
|
204 log.debug("preloading %s in %s" % (title, self._id)) |
243
63e9b680d3e7
browser_side, blog: better PEP8 compliance
souliane <souliane@mailoo.org>
parents:
165
diff
changeset
|
205 |
287
7a1dc69112b8
browser_side (plugin radiocol): send the current queue to new players
souliane <souliane@mailoo.org>
parents:
286
diff
changeset
|
206 def play(self, play=True): |
341
31c105017d6b
browser_side: radiocol current song playback will jump to time when a user joins a running session
souliane <souliane@mailoo.org>
parents:
340
diff
changeset
|
207 """Play or pause the song |
31c105017d6b
browser_side: radiocol current song playback will jump to time when a user joins a running session
souliane <souliane@mailoo.org>
parents:
340
diff
changeset
|
208 @param play: set to True to play or to False to pause |
31c105017d6b
browser_side: radiocol current song playback will jump to time when a user joins a running session
souliane <souliane@mailoo.org>
parents:
340
diff
changeset
|
209 """ |
287
7a1dc69112b8
browser_side (plugin radiocol): send the current queue to new players
souliane <souliane@mailoo.org>
parents:
286
diff
changeset
|
210 if play: |
7a1dc69112b8
browser_side (plugin radiocol): send the current queue to new players
souliane <souliane@mailoo.org>
parents:
286
diff
changeset
|
211 self.played = True |
7a1dc69112b8
browser_side (plugin radiocol): send the current queue to new players
souliane <souliane@mailoo.org>
parents:
286
diff
changeset
|
212 self.metadata.setTitle(self.title) |
7a1dc69112b8
browser_side (plugin radiocol): send the current queue to new players
souliane <souliane@mailoo.org>
parents:
286
diff
changeset
|
213 self.metadata.setArtist(self.artist) |
7a1dc69112b8
browser_side (plugin radiocol): send the current queue to new players
souliane <souliane@mailoo.org>
parents:
286
diff
changeset
|
214 self.metadata.setAlbum(self.album) |
341
31c105017d6b
browser_side: radiocol current song playback will jump to time when a user joins a running session
souliane <souliane@mailoo.org>
parents:
340
diff
changeset
|
215 Audio.play(self) |
287
7a1dc69112b8
browser_side (plugin radiocol): send the current queue to new players
souliane <souliane@mailoo.org>
parents:
286
diff
changeset
|
216 else: |
341
31c105017d6b
browser_side: radiocol current song playback will jump to time when a user joins a running session
souliane <souliane@mailoo.org>
parents:
340
diff
changeset
|
217 self.pause() |
129
dd0d39ae7d24
RadioCol: song preloading + fonctionnal players
Goffi <goffi@goffi.org>
parents:
128
diff
changeset
|
218 |
dd0d39ae7d24
RadioCol: song preloading + fonctionnal players
Goffi <goffi@goffi.org>
parents:
128
diff
changeset
|
219 |
127 | 220 class RadioColPanel(HorizontalPanel, ClickHandler): |
221 | |
285
4f0c2fea358a
browser_side (plugin radiocol): use the status label to give more information
souliane <souliane@mailoo.org>
parents:
284
diff
changeset
|
222 def __init__(self, parent, referee, player_nick, players, queue_data): |
397
6148e9063069
browser_side: radiocol displays who uploaded the file with ChatPanel.printInfo
souliane <souliane@mailoo.org>
parents:
394
diff
changeset
|
223 """ |
6148e9063069
browser_side: radiocol displays who uploaded the file with ChatPanel.printInfo
souliane <souliane@mailoo.org>
parents:
394
diff
changeset
|
224 @param parent |
6148e9063069
browser_side: radiocol displays who uploaded the file with ChatPanel.printInfo
souliane <souliane@mailoo.org>
parents:
394
diff
changeset
|
225 @param referee |
6148e9063069
browser_side: radiocol displays who uploaded the file with ChatPanel.printInfo
souliane <souliane@mailoo.org>
parents:
394
diff
changeset
|
226 @param player_nick |
6148e9063069
browser_side: radiocol displays who uploaded the file with ChatPanel.printInfo
souliane <souliane@mailoo.org>
parents:
394
diff
changeset
|
227 @param players |
6148e9063069
browser_side: radiocol displays who uploaded the file with ChatPanel.printInfo
souliane <souliane@mailoo.org>
parents:
394
diff
changeset
|
228 @param queue_data: list of integers (queue to start, queue limit) |
6148e9063069
browser_side: radiocol displays who uploaded the file with ChatPanel.printInfo
souliane <souliane@mailoo.org>
parents:
394
diff
changeset
|
229 """ |
286
e76ec07be8e5
browser_side (plugin radiocol): the UI uses CaptionPanel
souliane <souliane@mailoo.org>
parents:
285
diff
changeset
|
230 # We need to set it here and not in the CSS :( |
e76ec07be8e5
browser_side (plugin radiocol): the UI uses CaptionPanel
souliane <souliane@mailoo.org>
parents:
285
diff
changeset
|
231 HorizontalPanel.__init__(self, Height="90px") |
127 | 232 ClickHandler.__init__(self) |
233 self._parent = parent | |
234 self.referee = referee | |
285
4f0c2fea358a
browser_side (plugin radiocol): use the status label to give more information
souliane <souliane@mailoo.org>
parents:
284
diff
changeset
|
235 self.queue_data = queue_data |
127 | 236 self.setStyleName("radiocolPanel") |
243
63e9b680d3e7
browser_side, blog: better PEP8 compliance
souliane <souliane@mailoo.org>
parents:
165
diff
changeset
|
237 |
127 | 238 # Now we set up the layout |
239 self.metadata_panel = MetadataPanel() | |
286
e76ec07be8e5
browser_side (plugin radiocol): the UI uses CaptionPanel
souliane <souliane@mailoo.org>
parents:
285
diff
changeset
|
240 self.add(CaptionPanel("Now playing", self.metadata_panel)) |
e76ec07be8e5
browser_side (plugin radiocol): the UI uses CaptionPanel
souliane <souliane@mailoo.org>
parents:
285
diff
changeset
|
241 self.playlist_panel = VerticalPanel() |
e76ec07be8e5
browser_side (plugin radiocol): the UI uses CaptionPanel
souliane <souliane@mailoo.org>
parents:
285
diff
changeset
|
242 self.add(CaptionPanel("Songs queue", self.playlist_panel)) |
285
4f0c2fea358a
browser_side (plugin radiocol): use the status label to give more information
souliane <souliane@mailoo.org>
parents:
284
diff
changeset
|
243 self.control_panel = ControlPanel(self) |
286
e76ec07be8e5
browser_side (plugin radiocol): the UI uses CaptionPanel
souliane <souliane@mailoo.org>
parents:
285
diff
changeset
|
244 self.add(CaptionPanel("Controls", self.control_panel)) |
e76ec07be8e5
browser_side (plugin radiocol): the UI uses CaptionPanel
souliane <souliane@mailoo.org>
parents:
285
diff
changeset
|
245 |
130 | 246 self.next_songs = [] |
287
7a1dc69112b8
browser_side (plugin radiocol): send the current queue to new players
souliane <souliane@mailoo.org>
parents:
286
diff
changeset
|
247 self.players = [Player("player_%d" % i, self.metadata_panel) for i in xrange(queue_data[1] + 1)] |
129
dd0d39ae7d24
RadioCol: song preloading + fonctionnal players
Goffi <goffi@goffi.org>
parents:
128
diff
changeset
|
248 self.current_player = None |
dd0d39ae7d24
RadioCol: song preloading + fonctionnal players
Goffi <goffi@goffi.org>
parents:
128
diff
changeset
|
249 for player in self.players: |
286
e76ec07be8e5
browser_side (plugin radiocol): the UI uses CaptionPanel
souliane <souliane@mailoo.org>
parents:
285
diff
changeset
|
250 self.add(player) |
127 | 251 self.addClickListener(self) |
130 | 252 |
397
6148e9063069
browser_side: radiocol displays who uploaded the file with ChatPanel.printInfo
souliane <souliane@mailoo.org>
parents:
394
diff
changeset
|
253 help_msg = """Accepted file formats: Ogg Vorbis (recommended), MP3.<br /> |
394
ee61b0765d6c
browser_side: radiocol supports MP3
souliane <souliane@mailoo.org>
parents:
370
diff
changeset
|
254 Please do not submit files that are protected by copyright.<br /> |
397
6148e9063069
browser_side: radiocol displays who uploaded the file with ChatPanel.printInfo
souliane <souliane@mailoo.org>
parents:
394
diff
changeset
|
255 Click <a style="color: red;">here</a> if you need some support :)""" |
6148e9063069
browser_side: radiocol displays who uploaded the file with ChatPanel.printInfo
souliane <souliane@mailoo.org>
parents:
394
diff
changeset
|
256 link_cb = lambda: self._parent.host.bridge.call('joinMUC', None, DEFAULT_MUC, self._parent.nick) |
6148e9063069
browser_side: radiocol displays who uploaded the file with ChatPanel.printInfo
souliane <souliane@mailoo.org>
parents:
394
diff
changeset
|
257 self._parent.printInfo(help_msg, type_='link', link_cb=link_cb) |
284
bee4719af9b9
browser_side (plugin radiocol): info message when you start the radiocol
souliane <souliane@mailoo.org>
parents:
283
diff
changeset
|
258 |
130 | 259 def pushNextSong(self, title): |
260 """Add a song to the left panel's next songs queue""" | |
261 next_song = Label(title) | |
262 next_song.setStyleName("radiocol_next_song") | |
263 self.next_songs.append(next_song) | |
286
e76ec07be8e5
browser_side (plugin radiocol): the UI uses CaptionPanel
souliane <souliane@mailoo.org>
parents:
285
diff
changeset
|
264 self.playlist_panel.append(next_song) |
287
7a1dc69112b8
browser_side (plugin radiocol): send the current queue to new players
souliane <souliane@mailoo.org>
parents:
286
diff
changeset
|
265 self.control_panel.updateStatus() |
130 | 266 |
267 def popNextSong(self): | |
268 """Remove the first song of next songs list | |
269 should be called when the song is played""" | |
270 #FIXME: should check that the song we remove is the one we play | |
271 next_song = self.next_songs.pop(0) | |
286
e76ec07be8e5
browser_side (plugin radiocol): the UI uses CaptionPanel
souliane <souliane@mailoo.org>
parents:
285
diff
changeset
|
272 self.playlist_panel.remove(next_song) |
287
7a1dc69112b8
browser_side (plugin radiocol): send the current queue to new players
souliane <souliane@mailoo.org>
parents:
286
diff
changeset
|
273 self.control_panel.updateStatus() |
243
63e9b680d3e7
browser_side, blog: better PEP8 compliance
souliane <souliane@mailoo.org>
parents:
165
diff
changeset
|
274 |
285
4f0c2fea358a
browser_side (plugin radiocol): use the status label to give more information
souliane <souliane@mailoo.org>
parents:
284
diff
changeset
|
275 def getQueueSize(self): |
286
e76ec07be8e5
browser_side (plugin radiocol): the UI uses CaptionPanel
souliane <souliane@mailoo.org>
parents:
285
diff
changeset
|
276 return len(self.playlist_panel.getChildren()) |
285
4f0c2fea358a
browser_side (plugin radiocol): use the status label to give more information
souliane <souliane@mailoo.org>
parents:
284
diff
changeset
|
277 |
287
7a1dc69112b8
browser_side (plugin radiocol): send the current queue to new players
souliane <souliane@mailoo.org>
parents:
286
diff
changeset
|
278 def radiocolCheckPreload(self, timestamp): |
7a1dc69112b8
browser_side (plugin radiocol): send the current queue to new players
souliane <souliane@mailoo.org>
parents:
286
diff
changeset
|
279 for player in self.players: |
7a1dc69112b8
browser_side (plugin radiocol): send the current queue to new players
souliane <souliane@mailoo.org>
parents:
286
diff
changeset
|
280 if player.timestamp == timestamp: |
7a1dc69112b8
browser_side (plugin radiocol): send the current queue to new players
souliane <souliane@mailoo.org>
parents:
286
diff
changeset
|
281 return False |
7a1dc69112b8
browser_side (plugin radiocol): send the current queue to new players
souliane <souliane@mailoo.org>
parents:
286
diff
changeset
|
282 return True |
7a1dc69112b8
browser_side (plugin radiocol): send the current queue to new players
souliane <souliane@mailoo.org>
parents:
286
diff
changeset
|
283 |
397
6148e9063069
browser_side: radiocol displays who uploaded the file with ChatPanel.printInfo
souliane <souliane@mailoo.org>
parents:
394
diff
changeset
|
284 def radiocolPreload(self, timestamp, filename, title, artist, album, sender): |
287
7a1dc69112b8
browser_side (plugin radiocol): send the current queue to new players
souliane <souliane@mailoo.org>
parents:
286
diff
changeset
|
285 if not self.radiocolCheckPreload(timestamp): |
7a1dc69112b8
browser_side (plugin radiocol): send the current queue to new players
souliane <souliane@mailoo.org>
parents:
286
diff
changeset
|
286 return # song already preloaded |
129
dd0d39ae7d24
RadioCol: song preloading + fonctionnal players
Goffi <goffi@goffi.org>
parents:
128
diff
changeset
|
287 preloaded = False |
dd0d39ae7d24
RadioCol: song preloading + fonctionnal players
Goffi <goffi@goffi.org>
parents:
128
diff
changeset
|
288 for player in self.players: |
dd0d39ae7d24
RadioCol: song preloading + fonctionnal players
Goffi <goffi@goffi.org>
parents:
128
diff
changeset
|
289 if not player.filename or \ |
dd0d39ae7d24
RadioCol: song preloading + fonctionnal players
Goffi <goffi@goffi.org>
parents:
128
diff
changeset
|
290 (player.played and player != self.current_player): |
dd0d39ae7d24
RadioCol: song preloading + fonctionnal players
Goffi <goffi@goffi.org>
parents:
128
diff
changeset
|
291 #if player has no file loaded, or it has already played its song |
dd0d39ae7d24
RadioCol: song preloading + fonctionnal players
Goffi <goffi@goffi.org>
parents:
128
diff
changeset
|
292 #we use it to preload the next one |
287
7a1dc69112b8
browser_side (plugin radiocol): send the current queue to new players
souliane <souliane@mailoo.org>
parents:
286
diff
changeset
|
293 player.preload(timestamp, filename, title, artist, album) |
129
dd0d39ae7d24
RadioCol: song preloading + fonctionnal players
Goffi <goffi@goffi.org>
parents:
128
diff
changeset
|
294 preloaded = True |
dd0d39ae7d24
RadioCol: song preloading + fonctionnal players
Goffi <goffi@goffi.org>
parents:
128
diff
changeset
|
295 break |
dd0d39ae7d24
RadioCol: song preloading + fonctionnal players
Goffi <goffi@goffi.org>
parents:
128
diff
changeset
|
296 if not preloaded: |
439
d52f529a6d42
browser side: use of new log system (first draft):
Goffi <goffi@goffi.org>
parents:
422
diff
changeset
|
297 log.warning("Can't preload song, we are getting too many songs to preload, we shouldn't have more than %d at once" % self.queue_data[1]) |
130 | 298 else: |
299 self.pushNextSong(title) | |
397
6148e9063069
browser_side: radiocol displays who uploaded the file with ChatPanel.printInfo
souliane <souliane@mailoo.org>
parents:
394
diff
changeset
|
300 self._parent.printInfo(_('%(user)s uploaded %(artist)s - %(title)s') % {'user': sender, 'artist': artist, 'title': title}) |
243
63e9b680d3e7
browser_side, blog: better PEP8 compliance
souliane <souliane@mailoo.org>
parents:
165
diff
changeset
|
301 |
130 | 302 def radiocolPlay(self, filename): |
287
7a1dc69112b8
browser_side (plugin radiocol): send the current queue to new players
souliane <souliane@mailoo.org>
parents:
286
diff
changeset
|
303 found = False |
130 | 304 for player in self.players: |
287
7a1dc69112b8
browser_side (plugin radiocol): send the current queue to new players
souliane <souliane@mailoo.org>
parents:
286
diff
changeset
|
305 if not found and player.filename == filename: |
130 | 306 player.play() |
307 self.popNextSong() | |
308 self.current_player = player | |
287
7a1dc69112b8
browser_side (plugin radiocol): send the current queue to new players
souliane <souliane@mailoo.org>
parents:
286
diff
changeset
|
309 found = True |
7a1dc69112b8
browser_side (plugin radiocol): send the current queue to new players
souliane <souliane@mailoo.org>
parents:
286
diff
changeset
|
310 else: |
7a1dc69112b8
browser_side (plugin radiocol): send the current queue to new players
souliane <souliane@mailoo.org>
parents:
286
diff
changeset
|
311 player.play(False) # in case the previous player was not sync |
7a1dc69112b8
browser_side (plugin radiocol): send the current queue to new players
souliane <souliane@mailoo.org>
parents:
286
diff
changeset
|
312 if not found: |
439
d52f529a6d42
browser side: use of new log system (first draft):
Goffi <goffi@goffi.org>
parents:
422
diff
changeset
|
313 log.error("Song not found in queue, can't play it. This should not happen") |
127 | 314 |
130 | 315 def radiocolNoUpload(self): |
316 self.control_panel.blockUpload() | |
129
dd0d39ae7d24
RadioCol: song preloading + fonctionnal players
Goffi <goffi@goffi.org>
parents:
128
diff
changeset
|
317 |
130 | 318 def radiocolUploadOk(self): |
319 self.control_panel.unblockUpload() | |
320 | |
321 def radiocolSongRejected(self, reason): | |
322 Window.alert("Song rejected: %s" % reason) |