Compare commits

..

1167 Commits
0.5.5 ... 0.7.2

Author SHA1 Message Date
Jonas Kvinge
47b5dea95c Release 0.7.2 2020-08-15 23:01:56 +02:00
Jonas Kvinge
b0df63f1e8 Fix translations dir 2020-08-15 23:01:28 +02:00
Jonas Kvinge
3c4209b676 Add more compilation titles 2020-08-15 17:33:54 +02:00
Jonas Kvinge
d51b9a8e0e Add more compilation titles 2020-08-15 17:29:30 +02:00
Jonas Kvinge
3b56125bd2 Increase maximum time step for seeking to 120
Fixes #509
2020-08-15 15:18:58 +02:00
Jonas Kvinge
6e69e39007 Use static_cast instead for destroyed object 2020-08-15 15:16:06 +02:00
Jonas Kvinge
97208cb329 Add --auto to urpmi command for Mageia CI 2020-08-15 11:55:49 +02:00
Jonas Kvinge
414a4a97fb Use unicode option when replacing non-words
Fixes #513
2020-08-15 11:43:14 +02:00
Jonas Kvinge
2a809f96c4 Update Changelog 2020-08-15 11:31:29 +02:00
Jonas Kvinge
17799b03f3 Fix installation directory for translations
Fixes #512
2020-08-15 11:08:47 +02:00
Jonas Kvinge
efc55fc648 Fix typo in snapcraft.yaml 2020-08-15 02:36:54 +02:00
Jonas Kvinge
22bd41211a Turn back git revision 2020-08-15 02:36:37 +02:00
Jonas Kvinge
4cb0171bd0 Release 0.7.1 2020-08-15 00:44:28 +02:00
Strawbs Bot
ce6c5af72c Update translations 2020-08-15 00:25:52 +02:00
Jonas Kvinge
171575256c Remove broken iPhone (libimobiledevice) support
Fixes #212
2020-08-14 21:38:08 +02:00
Jonas Kvinge
d3664dcf78 Set QNetworkRequest::RedirectPolicyAttribute with Qt >= 5.9 2020-08-14 20:31:04 +02:00
Jonas Kvinge
0788981783 Set QNetworkRequest::RedirectPolicyAttribute with Qt >= 5.9 2020-08-14 20:20:41 +02:00
Jonas Kvinge
aeee7c02d5 Update Changelog 2020-08-14 18:39:56 +02:00
Jonas Kvinge
6e49a50461 Update Changelog 2020-08-14 18:38:50 +02:00
Jonas Kvinge
fbc99827ab Revert "Turn off sort indicators for playlist"
This reverts commit 7b50ec4630.
2020-08-14 17:30:27 +02:00
Jonas Kvinge
3b134320c4 Fix minor issue in cue parser with date and genre 2020-08-13 21:14:12 +02:00
Jonas Kvinge
c315e5016d Change mtime and ctime to qint64 2020-08-13 21:09:06 +02:00
Jonas Kvinge
7aebd6ed57 Only install translations if HAVE_TRANSLATIONS is set 2020-08-13 21:05:09 +02:00
Jonas Kvinge
1a8ca06495 Only install translations when INSTALL_TRANSLATIONS is set 2020-08-13 20:55:53 +02:00
Jonas Kvinge
a27ae7e4a6 Add CMake option to install translations
Fixes #485
2020-08-13 19:53:36 +02:00
Strawbs Bot
dd0ab897aa Update translations 2020-08-13 01:03:32 +02:00
Jonas Kvinge
00ad92fb6d Hide unavailable collection context menu actions 2020-08-12 21:34:42 +02:00
Jonas Kvinge
f84128ecbd Remove unused collection playlist container type 2020-08-12 21:33:38 +02:00
Jonas Kvinge
ba89e0f4e3 Fix MockPlaylistItem 2020-08-12 20:03:45 +02:00
Jonas Kvinge
832d36a4d4 Add Qt 6 option to strawberry.nsi 2020-08-12 18:07:21 +02:00
Jonas Kvinge
0b437b3bfb Use standard text color for links in about
Fixes #508
2020-08-12 17:27:08 +02:00
Jonas Kvinge
7b50ec4630 Turn off sort indicators for playlist
Fixes #511
2020-08-12 16:56:28 +02:00
Jonas Kvinge
4ddb13abac Increase maximum time step for seeking to 60
Fixes #509
2020-08-12 16:31:32 +02:00
Jonas Kvinge
d2ac081177 Fix taglib cmake type size checks 2020-08-11 16:23:07 +02:00
Strawbs Bot
9692fbf15b Update translations 2020-08-11 01:02:03 +02:00
Jonas Kvinge
be966488e8 Fix OSD D-Bus assertion with Qt 6 2020-08-10 23:05:07 +02:00
Jonas Kvinge
0ce613264f Make sure to always use original metadata when editing tags 2020-08-10 21:32:14 +02:00
Jonas Kvinge
34634d776e Make sure to always use original metadata when editing tags 2020-08-10 21:27:56 +02:00
Jonas Kvinge
673ded3819 Add proper check for collection song in edit tag dialog 2020-08-10 21:27:27 +02:00
Jonas Kvinge
b9a94ad3ae Default originalyear to -1 2020-08-10 18:06:20 +02:00
Jonas Kvinge
3f80b330cc Log artist and album name 2020-08-10 18:05:52 +02:00
Jonas Kvinge
01632d538c Decrease score for more compilation albums 2020-08-10 17:39:40 +02:00
Strawbs Bot
b0a9b1cd09 Update translations 2020-08-10 01:05:14 +02:00
Jonas Kvinge
1f772081fd Only update temporary metadata when set
Fixes #507
2020-08-10 00:32:57 +02:00
Jonas Kvinge
4ae54dbaad Decrease score for more compilation albums 2020-08-09 20:58:27 +02:00
Jonas Kvinge
e47f4ff731 Fix musixmatch cover size 2020-08-09 20:15:24 +02:00
Jonas Kvinge
465369d79e Base initial score on album cover sizes retrieved from API 2020-08-09 20:10:53 +02:00
Jonas Kvinge
15ddf6ff20 Save and restore playlist scrollbar position when switching between playlists 2020-08-09 14:00:56 +02:00
Jonas Kvinge
16a753bd95 Treat erors returned by the URL handler as non fatal
Fixes #505
2020-08-09 02:52:18 +02:00
Jonas Kvinge
c15103636c Fix NotificationPreview signal slot 2020-08-09 02:07:22 +02:00
Jonas Kvinge
8a5f82ee7d Tidal: Only return streamable songs in result
Fixes #505
2020-08-09 01:59:28 +02:00
Jonas Kvinge
5ec33ec821 Tidal: Show API error instead of network error when available 2020-08-09 01:50:03 +02:00
Jonas Kvinge
ab7d383cf1 Use virtual functions for OSD 2020-08-09 01:37:00 +02:00
Strawbs Bot
184e9a5c93 Update translations 2020-08-09 01:01:43 +02:00
Jonas Kvinge
c4f5363cde Properly enable/disable queue buttons depending on selection 2020-08-08 20:36:03 +02:00
Jonas Kvinge
002882cebf Build Qt 6 without linguist, qtbase code keeps updating breaking qt-tools 2020-08-08 19:20:27 +02:00
Jonas Kvinge
1cb3ec0c7b Only add autocomplete tags to playlist menu when we have chromaprint and gstreamer 2020-08-08 19:05:14 +02:00
Jonas Kvinge
6bf325c6f6 Fix QSslSocket::ignoreSslErrors compile error with Qt 6 2020-08-08 19:04:44 +02:00
Strawbs Bot
09c7ff9e8b Update translations 2020-08-08 01:01:46 +02:00
Jonas Kvinge
c2a94b61bf Fixes to playlist context menu
- Add all playlist actions to initialization list
- Make rescan songs work for non-collection songs by using playlist item reload
- Only show add to another playlist and remove from playlist when songs are selected
- Add some missing icons

Fixes #503
2020-08-07 22:13:02 +02:00
Jonas Kvinge
1db16232de Only show rescan songs for collection songs
Fixes #503
2020-08-07 21:18:48 +02:00
Jonas Kvinge
3da681a6b1 Add fatal error for missing protobuf compiler 2020-08-07 19:43:03 +02:00
Strawbs Bot
a79b3e7852 Update translations 2020-08-07 01:03:20 +02:00
Jonas Kvinge
b1099e6974 Handle metadata with tilde in title 2020-08-07 00:52:09 +02:00
Jonas Kvinge
4f6e06131c Change allow album cover search check 2020-08-07 00:28:46 +02:00
Jonas Kvinge
19f69e9e6c Allow cover search only using either artist, album or title 2020-08-07 00:18:31 +02:00
Jonas Kvinge
01481da773 Use Qt::QueuedConnection for cover fetcher 2020-08-06 23:55:44 +02:00
Jonas Kvinge
3e8f7e1cf1 Register CoverSearchStatistics metatype 2020-08-06 23:54:54 +02:00
Jonas Kvinge
5da69646f2 Add authentication for Qobuz cover provider 2020-08-06 22:57:44 +02:00
Jonas Kvinge
3cac01583b Add username password dialog 2020-08-06 22:54:21 +02:00
Jonas Kvinge
d16a26605e Fix updating playlist songs when there are multiple files with the same URL
Fixes #501
2020-08-06 21:40:42 +02:00
Jonas Kvinge
a4f692c788 Only show playlist add file(s) to transcoder when songs are selected 2020-08-06 18:37:17 +02:00
Jonas Kvinge
9f01206c57 Only show open in file browser when songs are selected 2020-08-06 18:36:52 +02:00
Jonas Kvinge
d34fc551ed Add playlist right click option to copy URL 2020-08-06 18:29:35 +02:00
Jonas Kvinge
7aa5f0d258 Only show delete and save playlist button when item is selected
Fixes #500
2020-08-06 16:00:03 +02:00
Jonas Kvinge
276a34bb66 Fix parsing Tidal track duration with Qt 6 2020-08-06 15:58:53 +02:00
Jonas Kvinge
0d820eda12 Remove diacritics in FTS search 2020-08-05 23:31:52 +02:00
Strawbs Bot
1991c1b677 Update translations 2020-08-05 01:02:22 +02:00
Jonas Kvinge
459404e3f0 Rename organise to organize
Prefer US spelling
2020-08-04 21:18:14 +02:00
Jonas Kvinge
badc623a3c Update Changelog 2020-08-03 21:53:31 +02:00
Jonas Kvinge
8e39f92cb7 Make album optional when reading scrobbles from cache 2020-08-03 21:50:26 +02:00
Jonas Kvinge
3ff4885973 Fix reading ASF comment tag 2020-08-03 21:03:14 +02:00
Jonas Kvinge
f9d45f7657 Fix reading and saving MP4 lyrics tag 2020-08-03 20:44:25 +02:00
Jonas Kvinge
789ff9df5c Add SUPublicEDKey Info.plist 2020-08-03 19:31:28 +02:00
Jonas Kvinge
a2064ed16b Always bundle libraries provided by homebrew
Fixes #343
2020-08-03 00:14:17 +02:00
Jonas Kvinge
b3d06c0868 Make macdeploy properly handle loader_path and libicudata 2020-08-02 17:02:28 +02:00
Jonas Kvinge
db1a6b3e38 Manually copy libicudata for macOS deploy
Fixes #498
2020-08-02 15:47:39 +02:00
Jonas Kvinge
8390237cc4 Fix Sparkle integration for macOS 2020-08-02 06:32:01 +02:00
Jonas Kvinge
9967eae7bb Decrease album cover score if artist doesn't match and cover isn't requested using album title 2020-08-02 04:34:15 +02:00
Jonas Kvinge
b3be7d1c6f Add CI for Qt 6 2020-08-02 04:19:39 +02:00
Jonas Kvinge
33ccb5dbb2 Remove duplicate check for X11 2020-08-02 04:18:40 +02:00
Strawbs Bot
5b90c0d695 Update translations 2020-08-02 01:07:53 +02:00
Jonas Kvinge
472a660239 Update README.md 2020-08-01 23:46:59 +02:00
Jonas Kvinge
ab67536d9a Prevent compilation and live albums from being picked before studio albums for album cover searches based on artist + song title 2020-08-01 23:17:35 +02:00
Jonas Kvinge
ee85fb3aec Use QString() on non-translated text in collection filter widget 2020-08-01 22:50:02 +02:00
Jonas Kvinge
214b6f4358 Use correct qt sparkle include for Qt 6 2020-08-01 03:41:48 +02:00
Jonas Kvinge
af0d092054 Use sparkle to check for updates on macOS and Windows 2020-08-01 03:37:16 +02:00
Jonas Kvinge
b07903c3e9 Register QVector<int> 2020-08-01 03:32:25 +02:00
Jonas Kvinge
ffa4c6bf09 Add SUFeedURL to Info.plist 2020-08-01 03:31:29 +02:00
Jonas Kvinge
b4125fa56c Remove create-dmg.sh script and use create-dmg directly from CMake 2020-08-01 03:31:01 +02:00
Jonas Kvinge
2c72302087 Install sparkle for macOS builds 2020-08-01 03:28:43 +02:00
Jonas Kvinge
0fa52bc64f Fix macdeploy script 2020-08-01 03:28:01 +02:00
Jonas Kvinge
f55a80b15a Use Q_UNUSED 2020-08-01 03:23:50 +02:00
Strawbs Bot
0735483321 Update translations 2020-08-01 01:01:36 +02:00
Jonas Kvinge
53fc2c7c21 Add extra safety for overwriting files for filesystem storage 2020-07-31 21:45:01 +02:00
Jonas Kvinge
f22133c3c5 Fix translations for Qt 6 2020-07-30 20:49:24 +02:00
Jonas Kvinge
2d5a6d6583 Use album artist for album repeat mode 2020-07-30 20:46:30 +02:00
Strawbs Bot
dc4adf2836 Update translations 2020-07-30 01:01:45 +02:00
Jonas Kvinge
dd6e254e4f Use quotes in collection query to allow special characters
Fixes #492
2020-07-29 21:41:35 +02:00
Jonas Kvinge
4c028c1659 Use position().toPoint() with Qt 6 2020-07-29 21:40:03 +02:00
Jonas Kvinge
d332a6777a Use QSortFilterProxyModel::filterRegularExpression only with Qt 6 2020-07-29 21:39:02 +02:00
Strawbs Bot
378251f229 Update translations 2020-07-29 01:01:35 +02:00
Strawbs Bot
b6b9b903ed Update translations 2020-07-28 01:01:37 +02:00
Strawbs Bot
143d68cfd5 Update translations 2020-07-27 01:01:53 +02:00
Jonas Kvinge
a31eac1426 Base warning for show in file browser on unique directories
Fixes #484
2020-07-26 15:10:00 +02:00
Jonas Kvinge
797196f7fc Register column aligment as int too 2020-07-26 15:05:00 +02:00
Chongo Bong
c9d0bc81dd simple addition of playlist actions to icon (#490)
very simple addition of playlist/playback actions to strawberry's icon. this re-introduces some nice little polish that was present in clementine's .desktop file.
2020-07-24 20:38:17 +02:00
Strawbs Bot
b5448ff607 Update translations 2020-07-22 01:01:41 +02:00
Jonas Kvinge
5ebd363d5d Fixes to last.fm scrobbling
- Start array notation for parameters at 0
- Correctly send trackNumber
2020-07-21 03:14:02 +02:00
Strawbs Bot
1d439e673e Update translations 2020-07-20 01:05:14 +02:00
Jonas Kvinge
0b7b7656b2 Replace use of QRegExp 2020-07-20 00:57:42 +02:00
Jonas Kvinge
eb270df835 Use std::bind in QtConcurrent::run() to fix compile with Qt 6 2020-07-19 22:43:58 +02:00
Jonas Kvinge
ff73dd2183 Partial revert commit af67de8 2020-07-19 19:07:12 +02:00
Strawbs Bot
e043a03eb6 Update translations 2020-07-19 15:23:03 +02:00
Jonas Kvinge
3cb4e8e373 Fix OSD Pretty margin 2020-07-19 04:09:34 +02:00
Jonas Kvinge
7e6de528b4 Update Last.fm error codes 2020-07-19 03:47:21 +02:00
Jonas Kvinge
df901c30ef Use QString() for html codes in about dialog 2020-07-19 03:46:41 +02:00
Jonas Kvinge
13856b33ec Fix playlist filter with Qt 5 2020-07-18 22:37:49 +02:00
Jonas Kvinge
a3a1c6f4c8 Fix saving playlist column alignment 2020-07-18 18:18:34 +02:00
Jonas Kvinge
638998a861 Replace QTimeLine::CurveShape with QEasingCurve 2020-07-18 17:53:14 +02:00
Jonas Kvinge
6e2ec89a05 Use QMouseEvent::pos() 2020-07-18 17:35:03 +02:00
Jonas Kvinge
af67de8aa6 Use lambdas for QtConcurrent::run instead of NewClosure 2020-07-18 16:28:39 +02:00
Jonas Kvinge
425dac478e Replace QProcess::error() with QProcess::errorOccurred() 2020-07-18 16:23:39 +02:00
Jonas Kvinge
b15c4ecd10 Fix check for context tab in TabSwitched
Broken with Qt 6
2020-07-18 15:52:36 +02:00
Jonas Kvinge
d7f88cf3a4 Register QItemSelection 2020-07-18 06:02:54 +02:00
Jonas Kvinge
5b7fbcd9b8 Remove debian stretch 2020-07-18 05:28:26 +02:00
Jonas Kvinge
7af64b0782 Fix QRegularExpressionMatch in FMPSParser 2020-07-18 05:02:34 +02:00
Jonas Kvinge
b84c70e811 Link Qt6::Core5Compat in libstrawberry-tagreader 2020-07-18 05:01:54 +02:00
Jonas Kvinge
f5b245c72d Add option to compile with Qt 6 2020-07-18 04:47:54 +02:00
Jonas Kvinge
4307183817 Fix utilities test 2020-07-18 04:32:37 +02:00
Jonas Kvinge
aeb32783d6 Only use QTextStream::setCodec() with Qt < 6 2020-07-18 04:27:21 +02:00
Jonas Kvinge
3927b3bf27 Remove QPainter::HighQualityAntialiasing 2020-07-18 04:26:19 +02:00
Jonas Kvinge
da9d2f9417 Replace QPalette::Background with QPalette::Window 2020-07-18 04:25:29 +02:00
Jonas Kvinge
1cec48e8f8 Use static_cast 2020-07-18 04:25:15 +02:00
Jonas Kvinge
f1105393da Replace QDateTime::toTime_t() with QDateTime::toSecsSinceEpoch() 2020-07-18 04:24:16 +02:00
Jonas Kvinge
dc7047e3c2 Use QLocale::LongFormat 2020-07-18 04:22:59 +02:00
Jonas Kvinge
08b2945623 Remove Qt 5.6 backward compatibility 2020-07-18 04:21:45 +02:00
Jonas Kvinge
cbcc223150 Replace QRegExp with QRegularExpression 2020-07-18 04:21:19 +02:00
Jonas Kvinge
9830f21e4a Use setContentsMargins() on layout 2020-07-18 04:20:20 +02:00
Jonas Kvinge
5f49567bf7 Make GlobalShortcut::nativeEventFilter compatible with Qt 6 2020-07-18 04:18:46 +02:00
Jonas Kvinge
6154ae7342 Move QDialogButtonBox signal/slot connect from UI file to class 2020-07-18 04:17:27 +02:00
Jonas Kvinge
978f3a3682 Add QSslError include 2020-07-18 04:16:31 +02:00
Jonas Kvinge
1f0961d574 Make MainWindow::nativeEvent compatible with Qt 6 2020-07-18 04:15:42 +02:00
Jonas Kvinge
a101252701 Make OSDPretty compatible with Qt 6 2020-07-18 04:15:19 +02:00
Jonas Kvinge
c96c29b1e3 Replace QRegExp with QRegularExpression 2020-07-18 04:14:51 +02:00
Jonas Kvinge
3b0fc180ff Make QListWidget::mimeData compatible with Qt 6 2020-07-18 04:13:53 +02:00
Jonas Kvinge
9b8bfdf33c Replace QPalette::Background with QPalette::Window 2020-07-18 04:12:50 +02:00
Jonas Kvinge
4328831fcd Use globalPosition() 2020-07-18 04:09:36 +02:00
Jonas Kvinge
e5b3df41e9 Replace QRegExp with QRegularExpression 2020-07-18 04:05:07 +02:00
Jonas Kvinge
cf5259e218 Add QActionGroup include 2020-07-18 03:54:52 +02:00
Jonas Kvinge
f24b6a520c Replace QDateTime::toTime_t() with QDateTime::toSecsSinceEpoch() 2020-07-18 03:53:30 +02:00
Jonas Kvinge
4140163ab2 Mark unused parameters 2020-07-17 16:36:24 +02:00
Jonas Kvinge
7afde0e93f Fix compile warning in qsearchfield_mac.mm 2020-07-17 16:35:57 +02:00
Jonas Kvinge
d27a571882 Ignore compile warning in SBSystemPreferences.h 2020-07-17 16:35:11 +02:00
Jonas Kvinge
1c70e3be25 Ignore format nonliteral in tutils.h 2020-07-17 16:34:37 +02:00
Jonas Kvinge
1819f64467 Disable deprecation warning for QMacCocoaViewContainer 2020-07-17 16:33:10 +02:00
Jonas Kvinge
9852e588c1 Change compile options 2020-07-17 16:32:37 +02:00
Jonas Kvinge
71a1ea481b Replace some uses of static_cast with qobject_cast 2020-07-17 01:32:07 +02:00
Jonas Kvinge
9e32f0d778 Silence some compile warnings with reinterpret cast 2020-07-16 22:46:31 +02:00
Jonas Kvinge
4478174dc2 Fix incorrectly mapped keys
Fixes #483
2020-07-16 22:28:35 +02:00
Jonas Kvinge
07553476d4 Remove xine 2020-07-16 00:59:46 +02:00
Jonas Kvinge
1773283456 Remove xine metronom.pts_per_smpls check 2020-07-16 00:44:41 +02:00
Jonas Kvinge
221ab51d90 Simply startup behaviour 2020-07-16 00:06:51 +02:00
Jonas Kvinge
43e0dd922b Hide settings that are unavailable on macOS in the settings 2020-07-16 00:05:49 +02:00
Jonas Kvinge
b3262652c3 Turn back git revision 2020-07-14 18:40:48 +02:00
Strawbs Bot
0cfa2b8c20 Update translations 2020-07-14 01:01:50 +02:00
Jonas Kvinge
c4600b2187 Release 0.6.13 2020-07-13 23:01:33 +02:00
Jonas Kvinge
89b15e3b57 Add taglib include dirs to unit test CMake file 2020-07-13 22:53:32 +02:00
Jonas Kvinge
07d8c3e770 Add .clang-format to excluded files in maketarball 2020-07-13 22:52:38 +02:00
Jonas Kvinge
c5ae64070e Add some include and link directories to unit test CMake
Fixes unit tests compile on macOS
2020-07-13 22:28:39 +02:00
Jonas Kvinge
2cb7c52147 Update README.md 2020-07-13 20:40:34 +02:00
Jonas Kvinge
c045b16213 Update README.md 2020-07-13 20:39:01 +02:00
Jonas Kvinge
b29387d409 More fancy tabbar fixes
- Only use custom size override for large and small sidebar modes
- Use scroll buttons by default
- Set elide text mode off because macOS has it on by default
- Set tooltip for top icon only mode
- Set icon sizes except for on macOS
2020-07-13 18:29:48 +02:00
Jonas Kvinge
614a09db1d Fix some compile warnings 2020-07-13 18:11:57 +02:00
Strawbs Bot
bcc5450aef Update translations 2020-07-13 01:02:37 +02:00
Jonas Kvinge
b0420a0566 Adjust width of settingspages 2020-07-13 00:57:50 +02:00
Jonas Kvinge
08d627bc9f Update Changelog 2020-07-13 00:39:33 +02:00
Jonas Kvinge
5b5f048faf Add missing taglib test files 2020-07-13 00:12:15 +02:00
Jonas Kvinge
212ebae043 Fix tab order in Appearance settings 2020-07-12 21:33:39 +02:00
Jonas Kvinge
a0c99df6b2 Make icon size for small tabbar configurable too 2020-07-12 21:02:29 +02:00
Jonas Kvinge
b0fabd7897 Adjust fancy tabbar size 2020-07-12 19:43:59 +02:00
Jonas Kvinge
938811f24f Fix sizes of horizontal modes and icon on top modes in fancy tabbar 2020-07-12 18:53:41 +02:00
Strawbs Bot
374d6e4d0f Update translations 2020-07-12 01:01:47 +02:00
Jonas Kvinge
5f54cdee0f Add new tagreader tests 2020-07-10 20:41:37 +02:00
Jonas Kvinge
c2bd452391 Fix playlist shuffle test, current index will never be at the end 2020-07-10 17:06:20 +02:00
Jonas Kvinge
d2bfc73b91 Only set playlist view item delegates once 2020-07-10 16:18:31 +02:00
Jonas Kvinge
42c62206c8 Move the currently playing song to the top when the playlist is manually shuffled
Fixes #304
2020-07-09 20:52:06 +02:00
Jonas Kvinge
4208b512f8 Remove some unused stuff in CMakeLists.txt 2020-07-09 20:09:07 +02:00
Strawbs Bot
3a3d5cd487 Update translations 2020-07-09 01:02:03 +02:00
Jonas Kvinge
e143a6d126 Enable playlist model tests 2020-07-08 20:52:50 +02:00
Jonas Kvinge
a99a19aa60 Fix broken playlist undostack when i.e. removing duplicates and unavailable songs 2020-07-08 20:43:46 +02:00
Jonas Kvinge
aba7d2b6a4 Fix mock_playlistitem.h 2020-07-08 18:22:12 +02:00
Jonas Kvinge
2fc6f835a8 Update sqlite test to use FTS5 instead of FTS3 2020-07-08 17:46:04 +02:00
Jonas Kvinge
27ce0e6d60 Add REQUIRED to find_package for GnuTLS 2020-07-08 17:29:58 +02:00
Strawbs Bot
05c70f2adb Update translations 2020-07-08 01:12:31 +02:00
Jonas Kvinge
904097b7b1 Add back option to use system taglib, add warning at the bottom 2020-07-07 23:44:04 +02:00
Jonas Kvinge
4e003c12a6 Add a taste of Strawbs background image 2020-07-07 23:02:24 +02:00
Jonas Kvinge
24f2cfb29f Fix playlist background image not loading 2020-07-07 22:21:07 +02:00
Jonas Kvinge
3f2b683fca Change pcmanfm-qt to pcmanfm 2020-07-07 18:28:47 +02:00
Strawbs Bot
83cee26d7d Update translations 2020-07-07 01:03:43 +02:00
Jonas Kvinge
c7df0c9b28 Use Q_UNUSED 2020-07-06 20:32:17 +02:00
Jonas Kvinge
26a8bfaac4 Open directory instead of file for PCManFM 2020-07-06 20:27:48 +02:00
Strawbs Bot
fc8a3be7d9 Update translations 2020-07-05 01:02:03 +02:00
Jonas Kvinge
b195c4033d Remove Ubuntu Eoan from CI 2020-07-04 11:34:47 +02:00
Jonas Kvinge
ab2732c55d Remove Fedora 33 from CI 2020-07-04 11:20:07 +02:00
Jonas Kvinge
7fd5c058be Fix buffer overflow when coverting S24LE 2020-07-04 01:55:36 +02:00
Jonas Kvinge
b22e8b4702 Convert S24LE and F32LE for analyzer
Fixes #155
2020-07-04 00:29:11 +02:00
Strawbs Bot
e2c2af3447 Update translations 2020-07-03 01:13:01 +02:00
Jonas Kvinge
4cd738ecb5 Properly calculate tab width 2020-07-02 04:24:55 +02:00
Strawbs Bot
e838470c26 Update translations 2020-07-02 01:01:57 +02:00
Jonas Kvinge
68e0bc40e1 Dont use fixed font size in fancy tabbar 2020-07-01 17:23:57 +02:00
Jonas Kvinge
9fc8bcdf62 Increase tab size 2020-07-01 14:32:30 +02:00
Jonas Kvinge
227b14a0b6 Make text in fancy tabbar wrap if too wide 2020-07-01 02:53:18 +02:00
Strawbs Bot
abcd184d5d Update translations 2020-06-30 01:02:57 +02:00
Jonas Kvinge
3fd9f4b0df Make fancy tabbar large mode icon size configurable 2020-06-29 03:03:04 +02:00
Jonas Kvinge
12ff3e963b Increase icon sizes in fancy tabbar large mode 2020-06-29 02:15:11 +02:00
Jonas Kvinge
558e392234 Make search field clear buttons follow icon size settings 2020-06-29 01:09:02 +02:00
Strawbs Bot
e747c2e263 Update translations 2020-06-29 01:03:02 +02:00
Jonas Kvinge
4d78b30e8c Make icon sizes configurable, increase default sizes for icons
Fixes #250
2020-06-28 18:36:48 +02:00
Strawbs Bot
9a1520d5e3 Update translations 2020-06-28 01:13:47 +02:00
Jonas Kvinge
04f3543424 Rename some layouts in mainwindow UI 2020-06-28 00:17:22 +02:00
Jonas Kvinge
eae287be37 Remove -Wno-bool-conversion 2020-06-27 03:00:25 +02:00
Jonas Kvinge
a0e698b058 Remove add_dependencies for singleapplication 2020-06-27 02:53:45 +02:00
Jonas Kvinge
cd2932adea taglib: Only match APE if MAC is found in the beginning of the file 2020-06-27 02:01:24 +02:00
Strawbs Bot
286b270592 Update translations 2020-06-27 01:13:13 +02:00
Jonas Kvinge
d1ebef2cc5 Remove DSF and DSDIFF from lists of supported audio formats 2020-06-27 00:58:37 +02:00
Jonas Kvinge
deb567fde7 Enable show in file browser based on local songs
Fixes #472
2020-06-27 00:52:26 +02:00
Jonas Kvinge
26586fb9ef Fix some includes in taglib 2020-06-27 00:06:02 +02:00
Jonas Kvinge
5f71a558b9 Adapt most changes from taglib2 2020-06-26 23:30:30 +02:00
Jonas Kvinge
08882639e0 Move SingleApplication includes to non-system includes 2020-06-26 23:27:38 +02:00
Jonas Kvinge
837ae2932f Add SYSTEM to system includes in target_include_directories 2020-06-26 23:26:04 +02:00
Jonas Kvinge
b51cc21140 Use override 2020-06-26 23:01:57 +02:00
Jonas Kvinge
f1115ba706 Remove explicit twice 2020-06-26 22:43:46 +02:00
Jonas Kvinge
dc36aee7ff Change some explicit usage 2020-06-26 22:41:38 +02:00
Jonas Kvinge
740f9581e6 Change 0 to nullptr 2020-06-26 22:06:23 +02:00
Jonas Kvinge
de0e3a7bc9 Use initialization list in tidalrequest 2020-06-26 22:05:07 +02:00
Jonas Kvinge
95e894fb4d Update README.md 2020-06-26 20:35:13 +02:00
Jonas Kvinge
faacb0d69f Subsonic: Dont strip words from title
Fixes #476
2020-06-26 17:35:05 +02:00
Jonas Kvinge
28b139e1b8 Disable wasapi plugin in nsi 2020-06-25 23:28:53 +02:00
Jonas Kvinge
39b1e0676b Fix memory include 2020-06-25 22:51:00 +02:00
Jonas Kvinge
c05fc5bd36 Update README.md 2020-06-25 14:12:54 +02:00
Jonas Kvinge
bbae9157fb Update README.md 2020-06-25 14:00:23 +02:00
Jonas Kvinge
fc5c63a869 Update README.md 2020-06-25 13:59:29 +02:00
Jonas Kvinge
e2e068f8e5 Remove .github/ISSUE_TEMPLATE/feature_request.md 2020-06-25 13:45:08 +02:00
Jonas Kvinge
43a068e30e Use QRandomGenerator 2020-06-22 15:22:14 +02:00
Jonas Kvinge
2beb4b8c6f taglib: Remove virtual functions workaround in audioproperties 2020-06-22 03:20:17 +02:00
Jonas Kvinge
ad4230f7fa taglib: Remove added explicit in tiostream 2020-06-22 03:11:58 +02:00
Jonas Kvinge
00369a2cfe taglib: Fix some formatting 2020-06-22 03:11:35 +02:00
Jonas Kvinge
248e487dd5 taglib: Remove unused functions + add virtual and explicit 2020-06-22 02:32:37 +02:00
Jonas Kvinge
3a3dc02a66 taglib: Rename Properties to AudioProperties 2020-06-22 00:49:25 +02:00
Jonas Kvinge
f49c47c20d Remove xine from NSI 2020-06-21 22:49:06 +02:00
Jonas Kvinge
417ef1a7a4 Add libgstwasapi.dll to debug release 2020-06-21 22:47:13 +02:00
Strawbs Bot
8e8889483f Update translations 2020-06-21 01:02:03 +02:00
Jonas Kvinge
b5818a9b62 Look for 2.0 module of libplist and libusbmuxd
Fixes #467
2020-06-20 17:39:43 +02:00
Jonas Kvinge
e709f676dc Remove unused tooltip files 2020-06-20 17:31:36 +02:00
Jonas Kvinge
bb611bf655 Dont use HTML in tooltip on KDE
Fixes #466
2020-06-20 17:30:37 +02:00
Strawbs Bot
24e8ca67c5 Update translations 2020-06-20 01:02:05 +02:00
Jonas Kvinge
0695941a77 Fix showing/hiding playing widget when playback is started while window is hidden 2020-06-19 17:32:08 +02:00
Strawbs Bot
7c71efd1b3 Update translations 2020-06-19 01:02:43 +02:00
Jonas Kvinge
5099aff5c3 Fix increasing play count when stop after track is enabled
Fixes #458
2020-06-18 22:18:49 +02:00
Jonas Kvinge
d3463250a9 Add on startup options to show maximized or minimized 2020-06-18 21:46:36 +02:00
Jonas Kvinge
3b58c02db0 Improve CMake files (#460) 2020-06-17 22:56:20 +02:00
Jonas Kvinge
64f0313d3c Remove ifdef's on slots in shortcutsettings page 2020-06-17 19:58:21 +02:00
Strawbs Bot
d3958e757b Update translations 2020-06-17 01:01:57 +02:00
Jonas Kvinge
651020388d Use override 2020-06-15 21:55:05 +02:00
Jonas Kvinge
72ede666d4 Replace use of C style casts 2020-06-15 17:59:02 +02:00
Jonas Kvinge
a68c249d4e Use parentheses for macro arguments 2020-06-15 00:33:47 +02:00
Jonas Kvinge
56caab4461 Remove redundant initialization 2020-06-15 00:11:52 +02:00
Jonas Kvinge
13b60351a6 Replace use of deprecated C++ headers 2020-06-14 23:54:18 +02:00
Jonas Kvinge
ad49d38e46 CI: Remove extra media for mageia 2020-06-14 23:20:40 +02:00
Jonas Kvinge
082c9097e4 Fix parameter name mispatches 2020-06-14 18:58:24 +02:00
Jonas Kvinge
2fbdb29ebc Replace 0 with nullptr 2020-06-14 17:02:47 +02:00
Jonas Kvinge
ef34dce4dc Additional manual formatting to taglib sources 2020-06-14 17:01:05 +02:00
Jonas Kvinge
577b7d8ec8 Use lengthInMilliseconds() 2020-06-14 16:08:18 +02:00
Jonas Kvinge
4ce099294c Format taglib sources 2020-06-13 19:02:42 +02:00
Strawbs Bot
72bff7fa35 Update translations 2020-06-13 01:02:48 +02:00
Jonas Kvinge
8c36b8803f Update .github/workflows/ccpp.yml 2020-06-12 19:22:29 +02:00
Strawbs Bot
c69993bd57 Update translations 2020-06-12 01:02:54 +02:00
Strawbs Bot
ea3e4982d2 Update translations 2020-06-10 01:01:56 +02:00
Strawbs Bot
9a5f6c7b9b Update translations 2020-06-09 01:02:35 +02:00
Jonas Kvinge
033d56a4b3 Use SetMinimumSize in about dialog 2020-06-08 23:40:15 +02:00
Jonas Kvinge
dfa684cccc Turn back git revision 2020-06-08 19:04:56 +02:00
Jonas Kvinge
b4e0f43dfc Update ccpp.yml 2020-06-07 23:59:31 +02:00
Jonas Kvinge
783cf7f1b0 Release 0.6.12 2020-06-07 23:49:11 +02:00
Jonas Kvinge
92d6fc3fad Change default grouping to album disc 2020-06-07 23:36:37 +02:00
Jonas Kvinge
436cdea4fd Add libgstfaac.dll 2020-06-06 15:08:30 +02:00
Jonas Kvinge
6b144b3b1f Add faac to nsi 2020-06-06 14:56:31 +02:00
Jonas Kvinge
7820082eb8 Add windows builder 2020-06-06 01:50:40 +02:00
Strawbs Bot
e732f921a3 Update translations 2020-06-06 01:02:52 +02:00
Jonas Kvinge
8b7c5d8585 Fix saving OSD pretty settings 2020-06-05 23:46:07 +02:00
Jonas Kvinge
9e959b189c Save settings when tabbar colors are changed 2020-06-05 23:39:22 +02:00
Jonas Kvinge
5c7a4cdc22 Fix deprecated use of QProcess::startDetached 2020-06-05 22:15:58 +02:00
Strawbs Bot
756e619e07 Update translations 2020-06-05 01:02:01 +02:00
Jonas Kvinge
eea6194b90 Add liborc-0.4-0.dll to nsi 2020-06-04 07:41:49 +02:00
Strawbs Bot
a09c1fa154 Update translations 2020-06-04 01:01:40 +02:00
Strawbs Bot
38c742328c Update translations 2020-05-31 01:01:44 +02:00
Jonas Kvinge
1d5db1446d Sort folders added from file view
Fixes #449
2020-05-30 21:59:55 +02:00
Jonas Kvinge
3f5f3d143f Fix compile without gstreamer 2020-05-30 21:39:16 +02:00
Jonas Kvinge
d297a7198a Update CI 2020-05-30 03:49:37 +02:00
Strawbs Bot
e5bd99dee4 Update translations 2020-05-30 01:04:25 +02:00
Jonas Kvinge
2720e13e88 Update libprotobuf DLL 2020-05-30 00:20:16 +02:00
Jonas Kvinge
0e9c1789ff Dont append disc to album title
Fixes #438
2020-05-29 20:38:19 +02:00
Jonas Kvinge
281cb10f84 Fix shadowing member 2020-05-29 18:30:27 +02:00
Jonas Kvinge
69a92ffe72 Remove mistankely added include 2020-05-29 18:30:09 +02:00
Jonas Kvinge
6447f159e5 Fix endl 2020-05-29 18:06:50 +02:00
Jonas Kvinge
94430883ad Use Qt::endl with Qt 5.14 and higher 2020-05-29 18:04:31 +02:00
Jonas Kvinge
046512eb3d Use Qt::endl 2020-05-29 17:47:26 +02:00
Jonas Kvinge
4479d97e90 Change use of Qt::KeyboardModifiers 2020-05-29 17:47:10 +02:00
Jonas Kvinge
bf5fea8951 Replace use of QMultiMap::insertMulti with QMultiMap::insert 2020-05-29 17:46:41 +02:00
Jonas Kvinge
07282e3de6 Change use of QLabel::pixmap 2020-05-29 17:45:00 +02:00
Jonas Kvinge
c2f90a20df Replace hex with Qt::hex 2020-05-29 17:43:44 +02:00
Jonas Kvinge
481d2d699e Replace QWheelEvent::delta with QWheelEvent::angleDelta 2020-05-29 17:42:40 +02:00
Jonas Kvinge
cf9a7e6ed3 Dont do qsrand on Qt versions higher than Qt 5.10 2020-05-29 17:40:34 +02:00
Jonas Kvinge
c35235371a Replace QString::SkipEmptyParts with Qt::SkipEmptyParts on Qt 5.14.0 or higher 2020-05-29 17:40:11 +02:00
Jonas Kvinge
5d5723ad58 Replace qrand with QRandomGenerator when using Qt 5.10 or higher 2020-05-29 17:37:46 +02:00
Jonas Kvinge
6c77294a86 Fixes to imobiledeviceconnection support 2020-05-29 17:36:01 +02:00
Strawbs Bot
823f65f1ca Add Czech 2020-05-29 01:52:17 +02:00
Strawbs Bot
0c378c1642 Update translations 2020-05-29 01:05:58 +02:00
Jonas Kvinge
bbb4162867 Make it possible to maximize console dialog 2020-05-26 18:27:01 +02:00
Jonas Kvinge
5dbdde3f2b Make scrobbler show error dialog for all errors when option is on 2020-05-26 17:51:23 +02:00
Jonas Kvinge
2521954bd9 Change layout name 2020-05-26 00:02:16 +02:00
Jonas Kvinge
5f1002894e Only save settings that has been changed 2020-05-25 23:56:54 +02:00
Jonas Kvinge
0489b312a3 Fix CircleCI 2020-05-25 23:56:29 +02:00
Jonas Kvinge
732be5a34f Use g_free 2020-05-24 23:22:18 +02:00
Jonas Kvinge
1f45c78ebb Unmap buffer references in error cases 2020-05-24 23:21:26 +02:00
Strawbs Bot
8b86c79bf9 Update translations 2020-05-18 01:01:33 +02:00
Jonas Kvinge
710ed81067 Require sqlite on centos too 2020-05-17 19:36:37 +02:00
Jonas Kvinge
24ac0e7b9b Fix CircleCI tumbleweed build 2020-05-17 18:34:25 +02:00
Jonas Kvinge
15b2bfbb29 Set library paths for Linux too when USE_BUNDLE is set 2020-05-17 17:01:35 +02:00
Jonas Kvinge
b7494eb381 Increase about dialog height 2020-05-17 16:25:18 +02:00
Strawbs Bot
27ac590250 Update translations 2020-05-17 01:01:49 +02:00
Jonas Kvinge
972076edab Move ClearDiskCache connect 2020-05-16 19:35:25 +02:00
Jonas Kvinge
bfa9a1eb8a Fix tests
Fixes #440
2020-05-16 19:17:06 +02:00
Jonas Kvinge
b0966f14e6 Only connect ClearPixmapDiskCache if app is set 2020-05-16 18:25:13 +02:00
Jonas Kvinge
37cf0c2fb6 Turn back git revision 2020-05-16 18:24:51 +02:00
Jonas Kvinge
4eb11c32b0 Release 0.6.11 2020-05-16 14:40:35 +02:00
Jonas Kvinge
25457bc09a Release 0.6.11 2020-05-16 14:39:53 +02:00
Jonas Kvinge
d5cfb5f733 Merge branch 'master' of github.com:strawberrymusicplayer/strawberry 2020-05-16 14:14:34 +02:00
Jonas Kvinge
79ba6e628e Use art id from the API as cover filename for Subsonic
Fixes #433
2020-05-16 14:13:22 +02:00
King_DuckZ
ef73add05a Warning fix on gcc 8.3.0 (#439)
Fixes warning:
assuming signed overflow does not occur when
simplifying conditional to constant [-Wstrict-overflow]

Signed-off-by: Michele Santullo <m.santullo@posteo.net>

Co-authored-by: Michele Santullo <m.santullo@posteo.net>
2020-05-16 13:34:42 +02:00
Strawbs Bot
ec3d11fb27 Update translations 2020-05-16 01:03:55 +02:00
Jonas Kvinge
3fbc7031b5 Update about dialog 2020-05-16 00:45:15 +02:00
Jonas Kvinge
40beb5e428 Update Changelog 2020-05-15 23:54:15 +02:00
Jonas Kvinge
0f608c8ef0 Update debian/copyright 2020-05-15 23:48:44 +02:00
Jonas Kvinge
8509cb4743 Spotify: Fix clearing access token 2020-05-15 23:36:01 +02:00
Jonas Kvinge
f4429e8c4a Make Musicbrainz cover provider respect rate limiting 2020-05-15 22:53:21 +02:00
Jonas Kvinge
e9e0829cdc Remove end dash from title 2020-05-15 22:15:52 +02:00
Strawbs Bot
93f0230423 Update translations 2020-05-15 01:03:01 +02:00
plonibarploni
f26a0df4a4 strip directory from OpenInFileManager command (#436) 2020-05-14 22:19:26 +02:00
Jonas Kvinge
c7d4624282 Update README.md 2020-05-14 22:12:55 +02:00
Jonas Kvinge
b03eee2a22 Update Changelog 2020-05-14 22:10:18 +02:00
Jonas Kvinge
7d4d72e706 Make Discogs provider respect rate limiting 2020-05-14 19:31:40 +02:00
Jonas Kvinge
e3c367984b Make it possible to receive SearchResults before SearchFinished 2020-05-14 19:30:29 +02:00
Jonas Kvinge
0ebfa10d32 Update details in playing widget 2020-05-14 19:29:34 +02:00
Jonas Kvinge
16d9a077f0 emit SearchCoverInProgress before SearchCoverAutomatically 2020-05-14 19:29:07 +02:00
Strawbs Bot
a9d8bbad42 Update translations 2020-05-14 01:03:53 +02:00
Jonas Kvinge
d78bb94af3 Fix Tidal OAuth login 2020-05-13 21:56:11 +02:00
Jonas Kvinge
b139c0a824 Dont use song count from backend for CDDA devices 2020-05-13 19:42:13 +02:00
Jonas Kvinge
5b0b924d34 Fix crash in CD songloader 2020-05-13 19:00:57 +02:00
Strawbs Bot
f75acf820c Update translations 2020-05-13 01:05:26 +02:00
Jonas Kvinge
43a47f33ac Dont link chromaprint unless its enabled
Fixes #432
2020-05-12 22:39:56 +02:00
Jonas Kvinge
fcea3a0877 Add option to scrobbler setting for showing login error
Fixes #430
2020-05-12 22:25:00 +02:00
Jonas Kvinge
a950ec3bd5 Adjust login state widget placement for covers and lyrics settings 2020-05-12 22:15:53 +02:00
Jonas Kvinge
e35501ff0a Delete remaining network replies and local redirct server in destructor 2020-05-12 21:28:42 +02:00
Jonas Kvinge
4bfad9dad8 Fix use of QString::right() 2020-05-12 21:12:08 +02:00
Jonas Kvinge
c5c7a07c12 Add QImageReader::imageFormatsForMimeType replacement function 2020-05-12 19:48:37 +02:00
Jonas Kvinge
7e22e0e552 Use original image format when saving images from Subsonic and Tidal
Fixes #435
2020-05-12 18:50:57 +02:00
Jonas Kvinge
84ec4bdc79 Check content type for image in album cover fetcher search 2020-05-12 18:47:32 +02:00
Jonas Kvinge
2bcad9b637 Do AddOrUpdateSongs in database thread 2020-05-12 18:45:24 +02:00
Jonas Kvinge
c8d5f03070 Dont use reference in AlbumSongsReplyReceived 2020-05-12 15:58:36 +02:00
Jonas Kvinge
168e101a5a Subsonic: Disconnect signal/slots 2020-05-12 15:55:13 +02:00
Jonas Kvinge
b4bc7333d9 Use album id as cover filename for Subsonic
Fixes #433
2020-05-12 15:53:15 +02:00
Strawbs Bot
e8b58c940e Update translations 2020-05-11 01:03:48 +02:00
Jonas Kvinge
ec7202e3f6 Use refresh token for ListenBrainz 2020-05-11 00:51:18 +02:00
Jonas Kvinge
9a740f7962 Change variable name 2020-05-11 00:49:54 +02:00
Jonas Kvinge
9210fdee0d Make spotify refresh login 2020-05-10 17:10:20 +02:00
Jonas Kvinge
d7661f0964 Fix possible crash in album cover fetcher 2020-05-10 16:54:14 +02:00
Jonas Kvinge
139e148912 Use shared_ptr for scrobbler cache items 2020-05-10 14:59:04 +02:00
Jonas Kvinge
1b8dedb4ed Clear access token when login is expired 2020-05-10 14:53:40 +02:00
Jonas Kvinge
5d6b0fa329 Reset last played song when playlist is finished 2020-05-10 13:08:29 +02:00
Jonas Kvinge
f35bbd89c9 Initialize QNetworkReply pointer 2020-05-10 12:56:12 +02:00
Jonas Kvinge
538a9e42f4 Remove these 2020-05-10 12:50:37 +02:00
Jonas Kvinge
623147dea7 Add Json cover provider class 2020-05-10 12:49:11 +02:00
Jonas Kvinge
dfecd0cd12 Show Json parse error 2020-05-10 12:48:48 +02:00
Jonas Kvinge
fe3af3a676 Clear albums on close in cover manager 2020-05-10 11:50:05 +02:00
Strawbs Bot
25f60331ed Update translations 2020-05-10 01:04:15 +02:00
Jonas Kvinge
d4860a3426 Use defaults from context UI 2020-05-09 18:37:43 +02:00
Jonas Kvinge
e7e77ed86b Add automatically search for album cover to context settings 2020-05-09 18:31:10 +02:00
Jonas Kvinge
dc80459c59 Remove debug print 2020-05-09 02:30:32 +02:00
Jonas Kvinge
2f2de59234 Fix AuthError function 2020-05-09 02:07:51 +02:00
Jonas Kvinge
7bccc21878 Add setting for cover providers 2020-05-09 01:48:08 +02:00
Strawbs Bot
40f9dafa44 Update translations 2020-05-09 01:04:23 +02:00
Jonas Kvinge
355d436d29 Sort settings pages 2020-05-08 20:25:02 +02:00
Jonas Kvinge
079b684388 Remove duplicate include 2020-05-08 20:17:33 +02:00
Jonas Kvinge
fd11f46d30 Add album cover provider from Musixmatch 2020-05-08 20:14:16 +02:00
Jonas Kvinge
cb7099199a Fix memory include 2020-05-08 18:47:55 +02:00
Jonas Kvinge
8566d91e89 Remove some unneeded includes, etc 2020-05-08 18:44:07 +02:00
Jonas Kvinge
f44ce49ea7 Add setting for lyric providers and add more providers
Fixes #335
2020-05-08 18:35:36 +02:00
Jonas Kvinge
6ef69f6b32 Format code 2020-05-08 18:34:33 +02:00
Strawbs Bot
f5983d5f10 Update translations 2020-05-07 01:03:30 +02:00
Jonas Kvinge
54cce5e089 Use album grouping function 2020-05-06 22:35:55 +02:00
Jonas Kvinge
4e4e596a1e Change some parameters to const 2020-05-06 22:26:29 +02:00
Jonas Kvinge
727a1f5ad1 Sort songs in collection by song title instead of track if previous
grouping is not the album.

Fixes #295
2020-05-06 22:14:59 +02:00
Jonas Kvinge
85fa86625b Fix infinite loop in stylesheetloader
Fixes #361
2020-05-06 21:43:44 +02:00
Jonas Kvinge
2c91877f83 Add option to show/hide sidebar
Fixes #393
2020-05-06 18:15:17 +02:00
Jonas Kvinge
7d1fac44e9 Update non collection songs with manually unset cover 2020-05-05 23:57:37 +02:00
Jonas Kvinge
2e34abfc0d Fix mpris:artUrl when using embedded cover
Fixes #426
2020-05-04 23:23:24 +02:00
Jonas Kvinge
81ba63e247 Turn on git revision 2020-05-02 15:18:39 +02:00
Strawbs Bot
8b11a65522 Update translations 2020-05-02 01:01:45 +02:00
Jonas Kvinge
7190ad1d15 Remove styles directory when uninstalling 2020-05-01 19:55:45 +02:00
Jonas Kvinge
1c9bae5df5 Update libnettle dll in nsi 2020-05-01 18:41:36 +02:00
Jonas Kvinge
cc7fd73916 Update libhogweed dll in nsi 2020-05-01 16:55:58 +02:00
Jonas Kvinge
6d8725f268 Release 0.6.10 2020-05-01 16:42:20 +02:00
Jonas Kvinge
373c7cdbc4 Update Changelog 2020-05-01 16:40:54 +02:00
Jonas Kvinge
a4855bb33b Decrease margin for top title in context a little 2020-05-01 16:40:29 +02:00
Jonas Kvinge
57c1358ded Ignore replies not containing images from Discogs 2020-05-01 12:02:15 +02:00
Jonas Kvinge
eb4ce1feab Use toUtf8() not toLocal8Bit() when converting string for UNC path
Fixes #418
2020-04-30 17:32:31 +02:00
Strawbs Bot
838c17e144 Update translations 2020-04-30 01:03:38 +02:00
Strawbs Bot
4499cbca3c Update translations 2020-04-29 01:08:07 +02:00
Jonas Kvinge
a835a4a2f7 Minor fixes to collection pixmap cache
- Add variables for cache size defaults
- Increase default disk cache size
- Change the pixmap cache settings UI to look better
- Add current pixmap disk cache used to settings
2020-04-29 00:33:38 +02:00
Jonas Kvinge
9cc6a94353 Replace some NewClosure's with lambda connects 2020-04-28 22:29:10 +02:00
Jonas Kvinge
5ed9d9c4a0 Update Changelog 2020-04-28 22:28:45 +02:00
Jonas Kvinge
9c5ac7080d Remove unused connect 2020-04-28 22:28:24 +02:00
Jonas Kvinge
6346370e86 Set some properties on font size spinboxes 2020-04-28 20:56:35 +02:00
Jonas Kvinge
80697f8f30 Add toolchain files to cmake dir 2020-04-28 16:24:40 +02:00
Jonas Kvinge
18b8b56367 Use leap 15.1 for source build 2020-04-28 16:23:27 +02:00
Jonas Kvinge
760aacca26 Make context fonts configurable
Fixes #362
2020-04-28 01:11:00 +02:00
Strawbs Bot
1a4f0dcf5a Update translations 2020-04-28 01:02:50 +02:00
Jonas Kvinge
947484a71c Set content margin for label top in context 2020-04-27 22:20:50 +02:00
Jonas Kvinge
a74439d038 Always load 32x32 icon for engine and device
Fixes #417
2020-04-27 21:36:49 +02:00
Jonas Kvinge
c338618593 Do size checking of icon sizes when loading system theme icons 2020-04-27 21:36:08 +02:00
Jonas Kvinge
011897da53 Remove gstreamer registry file on startup for Windows
Workaround for issue #266
2020-04-27 15:54:37 +02:00
Jonas Kvinge
3fcaa58947 Add back libtag.dll (required for GStreamer plugin) 2020-04-27 15:52:38 +02:00
Strawbs Bot
ef8bd4362a Update translations 2020-04-27 01:03:05 +02:00
Jonas Kvinge
627a2ef6dd Dont use system icon for clear search field icon
Fixes #413
2020-04-27 00:24:49 +02:00
Jonas Kvinge
2732536d6e Fix device state text color in devices
Fixes #414
2020-04-27 00:22:46 +02:00
Jonas Kvinge
5a1b4b3ff8 Remove logging include 2020-04-26 18:49:18 +02:00
Jonas Kvinge
d93ec82e4f Fix save album cover to file
Fixes #412
2020-04-26 18:48:07 +02:00
Jonas Kvinge
15080972f3 Turn off uniformItemSizes
Fixes #411
2020-04-26 18:32:42 +02:00
Strawbs Bot
171b58f737 Update translations 2020-04-26 01:09:51 +02:00
Jonas Kvinge
c008ab6141 Fix resume playback on startup for CUE 2020-04-25 15:57:02 +02:00
Jonas Kvinge
f14c3654dc Add search for lyrics as a seperate option
Double click album to show fullsize

Fixes #299
2020-04-25 14:48:43 +02:00
Jonas Kvinge
ae05a61551 Read date and genre from individual tracks in cue sheets
Fixes #347
2020-04-25 13:47:25 +02:00
Jonas Kvinge
8e1def225b Move some files 2020-04-25 01:59:21 +02:00
Jonas Kvinge
6e061764ee Remove phonon 2020-04-25 01:42:29 +02:00
Jonas Kvinge
ac55b22839 Fix scrobble duration 2020-04-25 01:15:23 +02:00
Jonas Kvinge
49f77d3b75 Change timeouts 2020-04-25 00:13:48 +02:00
Jonas Kvinge
4abc650edf Make scrobbler handle streams 2020-04-25 00:07:42 +02:00
Jonas Kvinge
5ba00b61be Remove unneeded includes 2020-04-25 00:07:18 +02:00
Jonas Kvinge
bc16a6c4cb Sort album cover search results by score and pick the first 3 2020-04-25 00:03:43 +02:00
Jonas Kvinge
ea4dc6f040 Add link directories 2020-04-24 19:48:57 +02:00
Strawbs Bot
749ae8d5eb Update translations 2020-04-24 01:10:40 +02:00
Jonas Kvinge
7a56ffb7c3 Use COMPILE_LANGUAGE when setting compile options 2020-04-24 00:01:59 +02:00
Jonas Kvinge
e62ab23de2 Add taglib includes 2020-04-24 00:01:23 +02:00
Jonas Kvinge
c6f6118506 Use system taglib on macOS 2020-04-24 00:00:07 +02:00
Jonas Kvinge
8a5d5ad952 Fix some compile warnings in taglib 2020-04-23 21:51:14 +02:00
Jonas Kvinge
49e2615d14 Fix missing declaration 2020-04-23 21:50:50 +02:00
Jonas Kvinge
9289394261 Remove some compile options 2020-04-23 21:50:25 +02:00
Jonas Kvinge
8da4c88fd3 Fix compile warnings 2020-04-23 21:08:28 +02:00
Jonas Kvinge
a303850341 Use initialization list in cueparser 2020-04-23 21:07:17 +02:00
Jonas Kvinge
fb33610672 Remove unneeded this 2020-04-23 21:06:54 +02:00
Jonas Kvinge
d024dd6563 Minor code fixes to Subsonic 2020-04-23 21:06:26 +02:00
Jonas Kvinge
0be48f9f59 Minor code fixes to Tidal 2020-04-23 21:05:57 +02:00
Jonas Kvinge
df9292bafe Remove unneeded this 2020-04-23 21:05:17 +02:00
Jonas Kvinge
c1dcef3477 Improve Musicbrainz cover provider 2020-04-23 21:04:37 +02:00
Jonas Kvinge
48bc1f8361 Improve Last.fm cover provider code 2020-04-23 21:03:36 +02:00
Jonas Kvinge
2b2b4dbcf4 Improve Discogs cover provider 2020-04-23 21:02:48 +02:00
Jonas Kvinge
a1eadecdef Fix compile warnings in tests 2020-04-23 21:01:34 +02:00
Jonas Kvinge
f0b529952d Fix some compile warnings in singleapplication 2020-04-23 21:00:43 +02:00
Jonas Kvinge
c1ac2debb8 Fix some compile warnings in taglib 2020-04-23 21:00:16 +02:00
Jonas Kvinge
ac40094d37 Update CMakeLists.txt files 2020-04-23 20:59:09 +02:00
Jonas Kvinge
cb2bb4cb67 Fix CI 2020-04-22 15:38:10 +02:00
Strawbs Bot
f2965940cc Update translations 2020-04-22 01:01:32 +02:00
Strawbs Bot
c9ca147898 Update translations 2020-04-21 01:08:13 +02:00
Jonas Kvinge
c379d7f846 Minor code improvements to Deezer cover provider 2020-04-20 23:52:06 +02:00
Jonas Kvinge
45ae1ed265 Make Tidal album cover provider search for tracks too 2020-04-20 23:26:36 +02:00
Jonas Kvinge
9bf00eff40 Minor changes to Qobuz cover provider (and fix compile) 2020-04-20 22:58:57 +02:00
Jonas Kvinge
1677b3d5b9 Add Qobuz album cover provider 2020-04-20 22:12:40 +02:00
Jonas Kvinge
2a6806004a Fix update song length in context 2020-04-20 18:52:59 +02:00
Jonas Kvinge
39347d69df Only show song length in context when available 2020-04-20 18:46:26 +02:00
Jonas Kvinge
a2c0e4d4b1 Improve album cover loader, lyrics search and streaming support
- Improve album cover loader
- Add album cover loader result struct
- Move album cover thumbnail scaling to album cover loader
- Make init art manual look for album cover images in song directory
- Make album cover search work for songs outside of collection and
  streams
- Make album cover search work based on artist + title if album is not
  present
- Update art manual in playlist for local files, devices and CDDA
- Make lyrics search work for streams
- Add stream dialog to menu
- Remove dead code in InternetSearchModel
- Simplify code in InternetSearchView
2020-04-20 18:03:18 +02:00
Jonas Kvinge
ab2ffd9ac1 Add configure internet service to menu 2020-04-20 18:01:45 +02:00
Jonas Kvinge
c69fff52cc Remove useless using std 2020-04-20 17:49:06 +02:00
Strawbs Bot
1cfe61dc72 Update translations 2020-04-20 01:01:39 +02:00
Strawbs Bot
a23f39d81e Update translations 2020-04-19 01:01:32 +02:00
Strawbs Bot
b7724ff583 Update translations 2020-04-18 01:01:40 +02:00
Jonas Kvinge
e5dba60fab Remove song_id and artist_id from initialization list 2020-04-17 22:17:57 +02:00
Jonas Kvinge
2ccf489a83 Remove debug line 2020-04-17 17:23:43 +02:00
Jonas Kvinge
068939ca0b Fallback to SHA1 hash for cover filename if artist / album is stripped 2020-04-17 17:22:50 +02:00
Strawbs Bot
94ba8614ec Update translations 2020-04-16 01:01:39 +02:00
Strawbs Bot
f12a0c2379 Update translations 2020-04-15 01:01:28 +02:00
Strawbs Bot
8ab257645b Update translations 2020-04-14 01:11:19 +02:00
Jonas Kvinge
6331a0615f Update device schema 2020-04-13 23:24:35 +02:00
Jonas Kvinge
a21855fa20 Merge pull request #406 from strawberrymusicplayer/tidal
Add back Tidal support
2020-04-13 22:37:41 +02:00
Jonas Kvinge
12150c2180 Change database file 2020-04-13 19:05:55 +02:00
Jonas Kvinge
d90aecb164 Add back Tidal support 2020-04-13 19:04:06 +02:00
Jonas Kvinge
e738f2bc9f Recreate indexes and views in upgrade schema 11 2020-04-13 16:32:31 +02:00
Jonas Kvinge
2f72c41cda Improve internet classes 2020-04-13 06:30:40 +02:00
Jonas Kvinge
aa43d42cdb Remove const from signal slot connects 2020-04-13 05:57:48 +02:00
Jonas Kvinge
be8228e33c Fix song_id check 2020-04-13 04:17:45 +02:00
Jonas Kvinge
5591472dbd Change artist and song ID to strings 2020-04-13 03:39:51 +02:00
Strawbs Bot
30e6ced4e9 Update translations 2020-04-13 01:01:36 +02:00
Strawbs Bot
5f4c2bae89 Update translations 2020-04-12 01:01:28 +02:00
Strawbs Bot
0f036d9a43 Update translations 2020-04-11 01:01:39 +02:00
Strawbs Bot
1695ac3a32 Update translations 2020-04-10 01:02:01 +02:00
Jonas Kvinge
07a19ba619 Use slash 2020-04-09 20:41:14 +02:00
Jonas Kvinge
4dd78d89a0 Add option to remove problematic filename characters 2020-04-09 19:59:31 +02:00
Jonas Kvinge
8f4056faa6 Allow all characters except slash and backslash when organising music
Fixes #404
2020-04-09 18:14:02 +02:00
Jonas Kvinge
7b40c33892 Add missing semicolon in desktop file 2020-04-09 16:02:06 +02:00
Jonas Kvinge
b06bb5142f Turn on git revision 2020-04-09 16:01:12 +02:00
Jonas Kvinge
d576156ee1 Update snap version 2020-04-09 03:41:08 +02:00
Jonas Kvinge
77079968a5 Release 0.6.9 2020-04-09 02:32:32 +02:00
Jonas Kvinge
e12aca6214 Update maketarball.sh.in 2020-04-09 02:32:17 +02:00
Strawbs Bot
2a72891f28 Update translations 2020-04-09 00:42:40 +02:00
Jonas Kvinge
6fe47e78f1 Remove APE file detection by content
Fixes #373
2020-04-09 00:26:25 +02:00
Jonas Kvinge
0094894b52 Update ccpp.yml 2020-04-08 23:57:59 +02:00
Jonas Kvinge
0951bfb7ac Update Changelog 2020-04-08 23:38:00 +02:00
Jonas Kvinge
60e8519b65 Fix track and title in playlist not being movable
Fixes #403
2020-04-08 22:44:39 +02:00
Jonas Kvinge
78df0ed707 Change variable name 2020-04-08 22:32:01 +02:00
Jonas Kvinge
e3da0d6c9f Update config.yml 2020-04-08 19:27:32 +02:00
Jonas Kvinge
5df30580a6 Switch GitHub actions to use openSUSE 2020-04-08 01:15:39 +02:00
Strawbs Bot
4816b7dcd8 Update translations 2020-04-08 01:09:21 +02:00
Jonas Kvinge
75bced198b Hide open Audio CD on Windows 2020-04-08 01:00:21 +02:00
Jonas Kvinge
b500813617 Disable open Audio CD on Windows 2020-04-08 00:56:19 +02:00
Jonas Kvinge
a80037f7fc Update .travis.yml 2020-04-08 00:42:00 +02:00
Jonas Kvinge
1a6345342f Add Travis-CI 2020-04-07 17:05:01 +02:00
Jonas Kvinge
21b2193cd0 Add explicit 2020-04-07 16:49:15 +02:00
Jonas Kvinge
3efc496c41 Add better error handling for CDDA loader 2020-04-07 16:48:12 +02:00
Strawbs Bot
16340fbb78 Add Hungarian 2020-04-07 02:02:31 +02:00
Jonas Kvinge
a858b28bc4 Remove unused QGuiApplication include 2020-04-07 01:44:03 +02:00
Jonas Kvinge
307961cc7e Center organise and transcoder dialog on same screen as mainwindow 2020-04-07 01:26:17 +02:00
Strawbs Bot
3074377b55 Update translations 2020-04-07 01:03:31 +02:00
Jonas Kvinge
5a3edc00ac Remove extra check for oversized window 2020-04-06 23:14:23 +02:00
Jonas Kvinge
5a5f50e1e4 Use current_screen() function in OSD Pretty 2020-04-06 23:01:50 +02:00
Jonas Kvinge
7f39a38d6c Center cover manager on same screen as mainwindow 2020-04-06 22:30:03 +02:00
Jonas Kvinge
8321a48af7 Add missing QGuiApplication include 2020-04-06 22:26:55 +02:00
Jonas Kvinge
7e613f032e Add upgrade step to GitHub Actions 2020-04-06 22:26:19 +02:00
Jonas Kvinge
1c38c39db2 Center settings on current screen 2020-04-06 22:02:32 +02:00
Jonas Kvinge
4a0235c2ed Check for null pointer in OSD Pretty 2020-04-06 22:01:44 +02:00
Jonas Kvinge
b92961fda0 Merge branch 'circleci' 2020-04-06 06:46:39 +02:00
Jonas Kvinge
a2a06be62f Merge pull request #399 from plonibarploni/patch-2
Fix 'Show in file browser' for Caja
2020-04-06 05:21:18 +02:00
plonibarploni
c0956ed3b0 Fix 'Show in file browser' for Caja 2020-04-05 22:33:06 -04:00
Jonas Kvinge
3705e4e4d4 Add Ubuntu Focal to CircleCI 2020-04-06 03:07:09 +02:00
Jonas Kvinge
3fdbe84573 Rewrite parts of context to be adjustable and adjust album to width 2020-04-06 02:47:57 +02:00
Strawbs Bot
fb8e7d803f Update translations 2020-04-06 01:01:36 +02:00
Strawbs Bot
b01821aa03 Add Polish
Fixes #397
2020-04-05 13:24:58 +02:00
Strawbs Bot
69f974c9d0 Update translations 2020-04-05 01:02:31 +02:00
Jonas Kvinge
5db8b743fe Subsonic: Fix setting size 2020-04-04 23:34:04 +02:00
Jonas Kvinge
c424b0c888 Subsonic: Check if int values are strings 2020-04-04 23:31:27 +02:00
Strawbs Bot
d035c7d3b8 Update translations 2020-04-01 01:01:37 +02:00
Jonas Kvinge
144ffbc428 Subsonic: Handle track as string 2020-04-01 00:33:00 +02:00
Jonas Kvinge
afa8486d40 Update bug_report.md 2020-03-22 20:33:46 +01:00
Jonas Kvinge
a4cd42c448 Update README.md 2020-03-22 19:39:24 +01:00
Jonas Kvinge
853f4b75fa Update README.md 2020-03-22 00:17:40 +01:00
Jonas Kvinge
b95612287f This is a combination of 2 commits. (#390)
Add CircleCI
2020-03-20 16:17:11 +01:00
Jonas Kvinge
64e779172c Remove Travis-CI from master branch 2020-03-20 00:33:05 +01:00
Jonas Kvinge
10a136b4a3 Update macdeploy.py 2020-03-19 21:54:02 +01:00
Jonas Kvinge
ddc3af28ab Add back Travis-CI for macOS builds 2020-03-19 19:21:24 +01:00
Strawbs Bot
f602948dcf Update translations 2020-03-19 01:03:27 +01:00
Jonas Kvinge
973229cf4e Dont allow copy music to optical drives
Fixes #387
2020-03-17 20:44:51 +01:00
Strawbs Bot
d1f5cbea4a Update translations 2020-03-17 01:01:55 +01:00
Jonas Kvinge
49fe670ddb Center text of track column in tag fetcher
Fixes #388
2020-03-16 23:30:53 +01:00
Strawbs Bot
d6d87675c9 Update translations 2020-03-16 01:02:00 +01:00
Jonas Kvinge
1ca1927904 Only search for cover automatically for collection songs 2020-03-15 02:56:22 +01:00
Jonas Kvinge
454c3d1a3c Fix macOS build 2020-03-15 02:00:13 +01:00
Jonas Kvinge
11f5004112 Use a shorter playlist name when songs are added from file view
Fixes #363
2020-03-15 01:56:48 +01:00
Jonas Kvinge
625343f698 Check xdg-mime on Unix to find default file manager on Unix
Fixes #382
2020-03-15 01:21:30 +01:00
Jonas Kvinge
89d96a3ec1 Update README.md 2020-03-12 19:36:07 +01:00
Jonas Kvinge
c4c47fa196 Update README.md 2020-03-12 19:28:42 +01:00
Jonas Kvinge
ca35894b82 Update README.md 2020-03-12 19:27:00 +01:00
Jonas Kvinge
0213b6556b Add missing memory include 2020-03-08 19:13:13 +01:00
Jonas Kvinge
4edfd9be30 Merge branch 'master' of github.com:strawberrymusicplayer/strawberry 2020-03-08 18:40:56 +01:00
Jonas Kvinge
e55d9cafb6 Update logger 2020-03-08 18:40:39 +01:00
Jonas Kvinge
8deb0ed556 Move setting highDPI options 2020-03-08 18:36:48 +01:00
Strawbs Bot
82f995f243 Add Korean 2020-03-07 15:26:13 +01:00
Jonas Kvinge
b41c2ca724 Update README.md 2020-03-07 15:19:11 +01:00
Strawbs Bot
06ddbaf2cc Update translations 2020-03-01 01:02:03 +01:00
Jonas Kvinge
2adbfe1fbd Update required Qt version in README 2020-02-29 21:50:59 +01:00
Jonas Kvinge
7ae049b559 Require Qt 5.6 or higher 2020-02-29 21:50:10 +01:00
Jonas Kvinge
72913ceb1a Check that Qt version is higher than 5.9 to use QDir::isEmpty() 2020-02-29 21:40:20 +01:00
Jonas Kvinge
6a2be22fa1 Remove empty directories when organizing music
Fixes #353
2020-02-28 22:23:12 +01:00
Strawbs Bot
eaf20ce8d2 Update translations 2020-02-26 01:07:17 +01:00
Jonas Kvinge
97f9e142b4 Set hiDPI options 2020-02-25 01:15:05 +01:00
Jonas Kvinge
2e0f7b367f Remove tidal and qobuz
Fixes #369
2020-02-25 01:08:03 +01:00
Jonas Kvinge
7312e3f452 Find backtrace quiet 2020-02-23 20:42:30 +01:00
Strawbs Bot
8faa9f075b Update translations 2020-02-23 01:02:20 +01:00
Jonas Kvinge
584dcae075 Change variables 2020-02-22 17:39:06 +01:00
Jonas Kvinge
910e869b8d Use selectedRows() where possible 2020-02-22 15:31:25 +01:00
Jonas Kvinge
469e00b396 Remove setBaseStyle() 2020-02-22 13:43:33 +01:00
Strawbs Bot
ec46c758ba Update translations 2020-02-22 01:01:59 +01:00
Jonas Kvinge
8970e46bce More code fixes to mainwindow 2020-02-22 00:09:45 +01:00
Jonas Kvinge
3a4107d903 Workaround crash when mapToSource() fails 2020-02-21 22:42:21 +01:00
Jonas Kvinge
52ee21d550 Remove C++11Compat.cmake 2020-02-18 01:22:04 +01:00
Strawbs Bot
cdda4826cb Update translations 2020-02-18 01:02:19 +01:00
Jonas Kvinge
6745df5041 Update libffi 2020-02-17 20:22:02 +01:00
Jonas Kvinge
3d3efe1117 Remove extra whitespace 2020-02-17 20:21:45 +01:00
Strawbs Bot
82e5ae05ce Update translations 2020-02-17 01:01:39 +01:00
Strawbs Bot
0dbdd55a6a Update translations 2020-02-12 01:09:03 +01:00
Jonas Kvinge
424b0e61cb Remove extra newlines 2020-02-12 00:07:05 +01:00
Jonas Kvinge
533da8f89c Use isLocalFile() 2020-02-12 00:06:19 +01:00
Strawbs Bot
31506662db Update translations 2020-02-10 01:08:12 +01:00
Jonas Kvinge
7ff1a88ca8 Add QCoreApplication to xineengine.cpp 2020-02-09 04:20:22 +01:00
Jonas Kvinge
1851f26e3f Reduce includes 2020-02-09 02:29:35 +01:00
Jonas Kvinge
84cd65dd6c Reduce includes 2020-02-08 15:03:11 +01:00
Jonas Kvinge
8e0d792bf0 Reduce includes 2020-02-08 03:40:30 +01:00
Strawbs Bot
e74548b991 Update translations 2020-02-08 01:04:53 +01:00
Jonas Kvinge
2356ff5ebb Fix tabs order
Fixes #366
2020-02-08 00:01:12 +01:00
Gavin D. Howard
691f5d99ca Implement disk caching of album art (#360)
* Implement disk caching of album art

This includes a button to clear the cache in the settings, as
requested.

Closes #358

* Make the cache size defaults match

* Implement the review by jonaski

* Fix more problems with the PR
2020-02-07 23:18:18 +01:00
Jonas Kvinge
ab7b65a30b Remove chartlyrics 2020-02-06 22:33:38 +01:00
Daniel Kolesa
814bb3006c libstrawberry-common: check for library/includes for backtrace() (#364)
This is a non-standard header, so it should be checked properly.
This fixes compilation with musl libc and possibly other C library
implementations.
2020-02-06 20:04:23 +01:00
Jonas Kvinge
4bd4d3ae03 Update ccpp.yml 2020-02-06 18:40:05 +01:00
Jonas Kvinge
af1b0ace19 Use bultin taglib on Windows too by default 2020-02-06 18:37:02 +01:00
Jonas Kvinge
fb54febe03 Update .gitignore 2020-02-06 18:36:34 +01:00
Strawbs Bot
28faae5a4a Update translations 2020-01-29 01:03:25 +01:00
Jonas Kvinge
624a920aec Dont update temporary metadata while editing song with inline editor 2020-01-28 19:41:46 +01:00
Jonas Kvinge
2bf8187bff Add missing signal 2020-01-28 18:19:15 +01:00
Jonas Kvinge
cf06bf91e3 Change git url 2020-01-24 17:27:29 +01:00
Jonas Kvinge
e96f0504c0 Move related stuff into Rpm/Deb Cmake files 2020-01-21 23:53:28 +01:00
Jonas Kvinge
5fd61725bd Fixes for debian packaging 2020-01-21 23:15:15 +01:00
Jonas Kvinge
3fddfa075b Update .gitignore 2020-01-21 23:14:40 +01:00
Strawbs Bot
d712e5c409 Update translations 2020-01-21 01:01:48 +01:00
Jonas Kvinge
6287afae6c Fix Subsonic compatibility with LMS
Fixes #354
2020-01-20 19:19:14 +01:00
Jonas Kvinge
925540055a Use QString::number 2020-01-20 18:15:37 +01:00
Jonas Kvinge
d4373aae93 Subsonic: Handle album id as string 2020-01-20 17:45:40 +01:00
Jonas Kvinge
c4be310081 Use static taglib on windows 2020-01-16 20:46:10 +01:00
Jonas Kvinge
d1aa276465 Move -DTAGLIB_STATIC 2020-01-16 01:56:47 +01:00
Jonas Kvinge
2af6fc7452 Add -DTAGLIB_STATIC 2020-01-16 01:40:30 +01:00
Jonas Kvinge
ed15394a96 Delete qwindowsvistastyle.dll on uninstall 2020-01-15 23:12:08 +01:00
Jonas Kvinge
3791d11162 Disable libgstwasapi.dll 2020-01-15 17:12:18 +01:00
Jonas Kvinge
032ddabc06 Add qwindowsvistastyle.dll to nsi 2020-01-15 01:27:14 +01:00
Jonas Kvinge
bfb93ac6ed Update alsa in snap 2020-01-12 18:42:50 +01:00
Jonas Kvinge
f6d889d752 Add qtwayland5 to snap
Fixes #344
2020-01-12 18:02:13 +01:00
Jonas Kvinge
a7053f4269 Update README.md 2020-01-12 17:02:46 +01:00
Jonas Kvinge
233a6d06be Update README.md 2020-01-12 17:02:19 +01:00
Jonas Kvinge
853224148f Remove macOS builds 2020-01-12 16:56:19 +01:00
Jonas Kvinge
c5c8c27b2e Update README.md 2020-01-12 16:34:04 +01:00
Jonas Kvinge
d352d5fcff Turn back git revision 2020-01-07 21:41:31 +01:00
Strawbs Bot
0003de9320 Update translations 2020-01-07 01:01:46 +01:00
Jonas Kvinge
04f43e77af Update Changelog 2020-01-05 23:49:34 +01:00
Jonas Kvinge
be7cc55488 Release 0.6.8 2020-01-05 23:27:31 +01:00
Jonas Kvinge
c8f3379a48 Fix crash when deleting playlist folder. 2020-01-05 23:26:07 +01:00
Jonas Kvinge
b5a7945e49 Use QModelIndex::model() 2020-01-05 23:25:23 +01:00
Strawbs Bot
0bdac2e97d Update translations 2020-01-05 22:52:27 +01:00
Jonas Kvinge
1468a821fb Fix restoring to correct screen when maximized 2020-01-05 22:21:55 +01:00
Jonas Kvinge
3cdc8dc4b6 Use QWidget::screen() with Qt 5.14 2020-01-05 19:15:28 +01:00
Jonas Kvinge
aa255aa7e6 Use current screen, not primary screen 2020-01-05 19:14:25 +01:00
Strawbs Bot
66e5ccb9cc Update translations 2020-01-05 01:02:24 +01:00
Jonas Kvinge
2215f300bf Added option to disable playlist clear button
Fixes #339
2020-01-04 06:38:25 +01:00
Jonas Kvinge
eec767406b Add confirmation before clearing playlists with more than 500 songs 2020-01-04 06:11:21 +01:00
Jonas Kvinge
31aa42c2fa Fix compile with translations on Windows 2020-01-03 02:07:37 +01:00
Strawbs Bot
e912c59402 Add Italian 2020-01-03 01:46:38 +01:00
Strawbs Bot
c9988976f3 Update translations 2020-01-03 01:02:34 +01:00
Jonas Kvinge
443be1c2c8 Increase lyrics score if lyrics text > 60 2020-01-02 19:21:27 +01:00
Jonas Kvinge
7f442cff3b Fix QProxyStyle 2020-01-02 18:57:53 +01:00
Strawbs Bot
3696ae44ad Update translations 2019-12-31 01:09:52 +01:00
Jonas Kvinge
fc2d601424 Remove useless stdbool.h include 2019-12-30 23:14:40 +01:00
Jonas Kvinge
8818f24114 Fix compile with Qt 5.14 and above 2019-12-30 02:28:54 +01:00
Jonas Kvinge
0e12c8249e Fix actions builds 2019-12-30 00:49:39 +01:00
Strawbs Bot
06d62a70a9 Update translations 2019-12-30 00:35:45 +01:00
Jonas Kvinge
76d8018ca2 Remove HTML from translations 2019-12-29 23:53:54 +01:00
Jonas Kvinge
4123b41a5e Merge branch 'master' of github.com:jonaski/strawberry 2019-12-29 23:47:07 +01:00
Jonas Kvinge
5930257fed Remove HTML from translations 2019-12-29 23:46:49 +01:00
Jonas Kvinge
d3d60327ab Remove the answer to life the universe and everything 2019-12-29 23:46:35 +01:00
Strawbs Bot
5080ffb9fc Update translations 2019-12-29 23:41:39 +01:00
Jonas Kvinge
9b688a5179 Remove HTML from translations
Fixes #260
2019-12-29 23:37:48 +01:00
Strawbs Bot
a603dc5227 Update translations 2019-12-29 01:02:01 +01:00
Jonas Kvinge
c25f682caf Emit ExitFinished if no internet services are enabled 2019-12-28 03:35:07 +01:00
Jonas Kvinge
27a2fd298d Add the device view container widget to the tabbar
Fixes bugs related to the tabbar and the widgets being unresponsive

Fixes #279
Fixes #321
2019-12-28 03:13:41 +01:00
Strawbs Bot
bb38053cb3 Update translations 2019-12-24 01:01:42 +01:00
Strawbs Bot
5e82ee8695 Add German 2019-12-22 14:56:55 +01:00
Strawbs Bot
6c691ff9a8 Update translations 2019-12-22 14:55:33 +01:00
Jonas Kvinge
ac5a14fe4a Add StartupWMClass to desktop file
Fixes #305
2019-12-22 14:16:24 +01:00
Jonas Kvinge
00402d13ef Fix collection watcher on macOS
Fixes #324
2019-12-22 14:15:25 +01:00
Gavin D. Howard
079a559247 Make context title and summary changeable (#329)
* Make context title and summary changeable

Closes #30

* Fix checkboxes on context settings page

So...I am new to Qt, and I forgot that checkboxes can have a label.
Duh. Fixed.

* Put context settings in a different place

* Put ReplaceMessage and ReplaceVariable in Utilities
2019-12-22 12:09:05 +01:00
Strawbs Bot
a19ea8fdba Update translations 2019-12-22 01:04:20 +01:00
Jonas Kvinge
c6e172f942 Remove unused afc_port_ variable 2019-12-22 00:54:56 +01:00
Jonas Kvinge
bdc9f3e8bf Use AFC_E_SUCCESS 2019-12-22 00:45:01 +01:00
Jonas Kvinge
8ec5a587fc Fix compile 2019-12-21 22:34:42 +01:00
Jonas Kvinge
9caf46f2fb Use QString::asprintf 2019-12-21 22:07:04 +01:00
Jonas Kvinge
13fdbfc5e8 Use QString::asprintf 2019-12-21 21:56:48 +01:00
Jonas Kvinge
882c94110e Update temporary metadata when tags are changed 2019-12-21 21:55:24 +01:00
Jonas Kvinge
97bc980611 Upgrade packages 2019-12-21 18:26:51 +01:00
Jonas Kvinge
be9bf5c173 Replace use of QSet::fromList with Qt 5.14 and above 2019-12-21 18:22:18 +01:00
Jonas Kvinge
6df38c389c Replace use of QSet::toList() with QSet::values() 2019-12-21 18:19:09 +01:00
Jonas Kvinge
6c6bceb8cc Replace use of QString::sprintf 2019-12-21 18:17:58 +01:00
Jonas Kvinge
b5a897bb4d Replace use of QString::sprintf 2019-12-21 18:15:45 +01:00
Jonas Kvinge
ab85b716bb Replace use of QTime::start() with QElapsedTimer 2019-12-21 18:11:54 +01:00
Jonas Kvinge
8e256e6d5c Fix indent 2019-12-21 17:55:24 +01:00
Jonas Kvinge
288036a63b Fix spelling 2019-12-21 17:54:43 +01:00
Jonas Kvinge
f2005c3343 Update README.md 2019-12-11 20:46:15 +01:00
Jonas Kvinge
79d516b899 Update README.md 2019-12-11 20:45:27 +01:00
Jonas Kvinge
28b2f6eae3 Update README.md 2019-12-11 20:21:03 +01:00
Strawbs Bot
09ed2b945c Update translations 2019-12-07 12:45:08 +01:00
Jonas Kvinge
74aec89ef0 Add French 2019-12-05 19:58:07 +01:00
Jonas Kvinge
4faa23d099 Add Indonesian 2019-12-05 19:57:14 +01:00
Jonas Kvinge
5c4997ab20 Turn back git revision 2019-12-05 19:53:41 +01:00
Jonas Kvinge
cd490ca946 Release 0.6.7 2019-11-27 23:09:49 +01:00
Jonas Kvinge
793f6b00e6 Allow 4 tagreader workers 2019-11-27 22:57:51 +01:00
Strawbs Bot
c1ec568fda Update translations 2019-11-27 01:03:22 +01:00
Jonas Kvinge
4e2b52975e Remove require for sqlite on centos since we use customized sqlite 2019-11-26 23:43:34 +01:00
Jonas Kvinge
c141df6b86 Remove low-latency setting for wasapisink 2019-11-26 22:30:56 +01:00
Jonas Kvinge
63b781765a Update copyrights 2019-11-26 22:30:14 +01:00
Jonas Kvinge
c121e141e7 Remove duplicate entries in Changelog 2019-11-26 22:29:13 +01:00
Jonas Kvinge
7544c9a9f5 Update nsi with libprotobuf-22.dll 2019-11-26 19:42:38 +01:00
Jonas Kvinge
722a088515 Only remove pixmap cache when removing parents in collection model 2019-11-26 19:42:05 +01:00
Jonas Kvinge
0ce75bff58 Remove unused code in context albums 2019-11-26 19:41:32 +01:00
Strawbs Bot
3744b6d3de Update translations 2019-11-26 01:02:03 +01:00
Jonas Kvinge
a4ebd91e8d Make sure QSqlQuery::exec() was successful 2019-11-25 22:43:09 +01:00
Jonas Kvinge
3bb0ee916d Update Changelog 2019-11-25 22:31:42 +01:00
Jonas Kvinge
fd6afdf5f3 Extend url test 2019-11-25 22:30:33 +01:00
Jonas Kvinge
47c13a840e Handle different urls in collection backend for backward compatibility 2019-11-25 22:29:12 +01:00
Jonas Kvinge
5b61992b3c Save cover urls encoded 2019-11-25 22:28:48 +01:00
Jonas Kvinge
36331dc253 Fix removing nodes from pending art 2019-11-25 22:25:29 +01:00
Strawbs Bot
4265cf31b2 Update translations 2019-11-25 01:05:00 +01:00
Jonas Kvinge
337e47269f Remove portable, we dont use it 2019-11-25 00:35:48 +01:00
Jonas Kvinge
7039234471 Fix compile collection model test 2019-11-25 00:35:16 +01:00
Jonas Kvinge
f7f9333d91 Add collection backend url tests 2019-11-25 00:34:59 +01:00
Jonas Kvinge
bf35665932 Update all songs for the same directory+album when updating compilations
- Fixes a bug where the songs are stuck in various artists, because the
album has child songs, it will be stuck with various artists as the
parent node.
2019-11-25 00:28:49 +01:00
Jonas Kvinge
089a2271c2 Add GitHub Actions 2019-11-24 20:04:05 +01:00
Jonas Kvinge
f8e83e3631 Fix loading replay gain setting
Fixes #311
2019-11-24 19:34:05 +01:00
Strawbs Bot
46fd329913 Update translations 2019-11-23 01:07:52 +01:00
Jonas Kvinge
6b9ba96e77 Revert "Tidal: Add explicit to album titles"
This reverts commit b7d360d850.
2019-11-21 22:55:39 +01:00
Jonas Kvinge
b7d360d850 Tidal: Add explicit to album titles 2019-11-20 22:13:28 +01:00
Jonas Kvinge
8e226302ab Allow scrobbling songs without album
Fixes #309
2019-11-20 21:30:41 +01:00
Jonas Kvinge
7795b9edaf Dont replace metadata when loading playlists 2019-11-20 19:34:57 +01:00
Jonas Kvinge
9375d9699a No 2019-11-19 21:51:15 +01:00
Jonas Kvinge
cf0442d5b8 Fix setting pixmap cache limit 2019-11-19 21:49:46 +01:00
Jonas Kvinge
b386ca14df Show fullsize cover on doubleclick 2019-11-19 21:20:36 +01:00
Jonas Kvinge
ea47fae31e Add seperator between "unset cover" and "show fullsize" 2019-11-19 21:19:44 +01:00
Jonas Kvinge
e0fed07b10 Change pixmap cache limit 2019-11-19 21:03:06 +01:00
Jonas Kvinge
779d5ff7b6 Dont reset pixmap cache on model reset 2019-11-19 20:56:03 +01:00
Jonas Kvinge
eb6fbd03ec Update Changelog 2019-11-19 20:49:54 +01:00
Jonas Kvinge
95409d1b0e Remove unused variables 2019-11-19 20:47:06 +01:00
Jonas Kvinge
49599c8731 Add back ChartLyrics
This reverts commit c992768efe.
2019-11-19 20:45:22 +01:00
Jonas Kvinge
572f94e000 Update Changelog 2019-11-18 17:25:57 +01:00
Jonas Kvinge
c0a2ad5f50 Change comment 2019-11-18 17:16:58 +01:00
Jonas Kvinge
71fa5acc74 Fix previous player and doubleclick playlist song behaviour settings 2019-11-17 23:46:10 +01:00
Jonas Kvinge
bac5b7679d Use killproc.exe instead in nsi 2019-11-17 16:34:30 +01:00
Strawbs Bot
93ade821a5 Update translations 2019-11-17 01:10:32 +01:00
Jonas Kvinge
a718e19979 Use KillProc in nsi 2019-11-15 23:25:12 +01:00
Jonas Kvinge
8ac83a46f5 Remove clang compiler flag 2019-11-15 23:24:37 +01:00
Jonas Kvinge
1b65dcd7df Fix comparison between signed/unsigned 2019-11-15 00:23:06 +01:00
Jonas Kvinge
bbad45f1e7 Minor cmake fixes 2019-11-15 00:22:41 +01:00
Jonas Kvinge
331b9cca18 Remove sudo 2019-11-14 21:23:50 +01:00
Jonas Kvinge
a9accb7d85 Refactor scrobbler authentication code
Fix a crash when authentication is cancelled
2019-11-14 21:07:30 +01:00
Jonas Kvinge
1862e70628 Declare song using source 2019-11-14 00:09:35 +01:00
Jonas Kvinge
c4f7054ca6 Use QUrl::FullyEncoded in update compilations 2019-11-13 23:51:04 +01:00
Jonas Kvinge
175e568a28 Minor improvements to update compilations 2019-11-13 21:27:04 +01:00
Jonas Kvinge
c8d580e7de No need to delete pixmap cache when deleting empty parents 2019-11-13 21:16:48 +01:00
Jonas Kvinge
bc0c50ee65 Remove commented code 2019-11-13 21:12:50 +01:00
Jonas Kvinge
45e9dd96d1 Remove left click on analyzer to popup menu
Fixes #294
2019-11-11 00:01:39 +01:00
Jonas Kvinge
1cc73562a3 Turn back git revision 2019-11-10 15:35:36 +01:00
Jonas Kvinge
d870ef0bd5 Release 0.6.6 2019-11-09 17:21:08 +01:00
Jonas Kvinge
53d308dac5 Exclude .github dir in maketarball.sh 2019-11-09 17:16:18 +01:00
Jonas Kvinge
89b06ae7c7 Mulitply samples by channels, dont hardcode to 2 2019-11-09 16:34:17 +01:00
Jonas Kvinge
d38485a8ad Update Changelog 2019-11-09 16:30:23 +01:00
Jonas Kvinge
834877c503 Refactor gstreamer engine code, equalizer and fix stereo balancer 2019-11-08 23:07:21 +01:00
Jonas Kvinge
d033b79af4 Remove exclusive for wasapisink
Fixes #283
2019-11-07 20:26:25 +01:00
Jonas Kvinge
daec2cc203 Remove g_type_init 2019-11-06 21:53:27 +01:00
Jonas Kvinge
4e593cebab Add const 2019-11-06 21:53:09 +01:00
Jonas Kvinge
73d7701e94 Update FUNDING.yml 2019-11-05 19:38:46 +01:00
Strawbs Bot
24f1d7a72f Update translations 2019-11-04 01:07:14 +01:00
Jonas Kvinge
6387a01d7b Fix updating compilations
Fixes #288
2019-11-03 23:23:04 +01:00
Jonas Kvinge
e838840548 Remove duplicate check 2019-11-03 19:56:10 +01:00
Jonas Kvinge
6a430b441e Remove debug line 2019-11-03 19:56:01 +01:00
Jonas Kvinge
7b977ea839 Rename EngineDevice --> DeviceFinders, Add MMDeviceFinder 2019-11-03 19:53:08 +01:00
Jonas Kvinge
62b8521cbe Update Changelog 2019-10-29 19:27:03 +01:00
Martin Delille
308244d901 Change email (#287)
I didn't noticed you mention me in the *About...* section. Thank you for that even if it is an oversized thank you for the contributions made! : sweat_smile:

I just update it to my personal address.
2019-10-29 19:12:29 +01:00
Jonas Kvinge
6f521183f9 Fix spelling 2019-10-28 20:20:13 +01:00
Jonas Kvinge
76c6f7e733 Update README.md 2019-10-28 20:06:22 +01:00
Strawbs Bot
80ebfbeb6b Update translations 2019-10-28 01:02:39 +01:00
Jonas Kvinge
793901b319 Revert accidental change to flac test file 2019-10-28 00:05:29 +01:00
Jonas Kvinge
3950df8ec9 Add libgstwasapi.dll to nsi 2019-10-27 23:55:03 +01:00
Jonas Kvinge
e800b236aa Simplify the pipeline
Fix issue where bitrate is updated incorrectly by stream discoverer
Fixes issue #282
Also make it possible to enable stereo balancer without enabling the
equalizer
2019-10-27 23:48:54 +01:00
Jonas Kvinge
4ab7871106 Add wasapisink to directsound devicefinder 2019-10-27 23:47:28 +01:00
Jonas Kvinge
3de85549b6 Add option to automatically select current playing track 2019-10-27 02:11:51 +01:00
Jonas Kvinge
73164f7182 Update scrobble point when song is restarted 2019-10-27 02:09:34 +01:00
Strawbs Bot
004b000890 Update translations 2019-10-21 01:23:16 +02:00
Jonas Kvinge
d9c703d944 Add gst/audio/audio.h include 2019-10-20 20:04:23 +02:00
Jonas Kvinge
364b650033 Convert S32LE to S16LE for analyzer 2019-10-20 18:52:58 +02:00
Jonas Kvinge
156eb874db Fix analyzer and cleanup old pipeline code
- Move HandoffCallback to audio queue
- Add new callback for detecting source format
- Remove old decodebin stuff
2019-10-20 02:56:47 +02:00
Strawbs Bot
1a28dd0311 Update translations 2019-10-20 01:24:47 +02:00
Jonas Kvinge
e29b4b8609 Add more alternative icon names 2019-10-20 01:11:40 +02:00
Jonas Kvinge
c02997e6d9 More icon fixes 2019-10-20 00:59:22 +02:00
Jonas Kvinge
7c9fc91af9 Enable system theme icons, add iconmapper and rename some icon names 2019-10-20 00:17:28 +02:00
Jonas Kvinge
cf5198ac64 Limit tagreader workers to 2 2019-10-19 15:09:18 +02:00
Strawbs Bot
e76ddd6dd2 Update translations 2019-10-19 14:56:24 +02:00
Jonas Kvinge
e3a4cf1cf5 Add option to prefer album artist when sending scrobbles
Fixes #274
2019-10-19 02:56:23 +02:00
Jonas Kvinge
5844616ea8 Only ignore closeEvent when minimizing to system tray
Fixes #277
2019-10-19 02:21:20 +02:00
Jonas Kvinge
abeb580228 Disable analyzer for other bit depths than 16
This removes the splitting of the pipeline with the tee.
Move HandoffCallback to the source, which makes it possible to convert the audio buffer in HandoffCallback later.
Until then just disable analyzer for other formats.

Removes tee and probe queue converter and sink
2019-10-19 01:45:24 +02:00
Ike Devolder
4d888dfce8 subsonic: change disc to discNumber (#278)
As found in the v1.13.0 xsd, the disc is named discNumber in the api,
currently there is a check for disc which will never be found in the
response.

@see http://www.subsonic.org/pages/inc/api/schema/subsonic-rest-api-1.13.0.xsd

Signed-off-by: BlackEagle <ike.devolder@gmail.com>
2019-10-17 17:06:30 +02:00
Jonas Kvinge
08ff6f0ede Only use gcc 2019-10-13 00:17:01 +02:00
Jonas Kvinge
c458c27231 Switch to opensuse leap 15.1 2019-10-12 23:49:41 +02:00
Jonas Kvinge
9821b70c38 Dont use gst_caps_to_string as it causes hang with some formats 2019-10-12 01:58:01 +02:00
Jonas Kvinge
8ab8401110 Update libprotobuf in nsi 2019-10-06 17:53:00 +02:00
Strawbs Bot
4f798c85cf Update translations 2019-10-05 01:01:44 +02:00
Jonas Kvinge
1949bdb43f Merge branch 'master' of github.com:jonaski/strawberry 2019-10-04 17:09:08 +02:00
Jonas Kvinge
8cf2e6b31b Remove all lines in settings 2019-10-04 17:08:35 +02:00
Jonas Kvinge
b1069f9f18 Disable use system icons option 2019-10-04 17:08:12 +02:00
Strawbs Bot
fb8dfa9ae9 Update translations 2019-10-04 01:07:02 +02:00
Jonas Kvinge
4402a56e94 Fix compile with optional components disabled 2019-10-03 23:29:52 +02:00
Jonas Kvinge
f449808ba3 Fix: imobiledevice depends on libgpod enabled 2019-10-03 22:54:40 +02:00
Jonas Kvinge
5599739433 Fix lowercased playlist album artist column
Fixes #275
2019-10-03 18:25:50 +02:00
Jonas Kvinge
197cf85f56 Turn back git revision 2019-10-01 21:42:08 +02:00
Jonas Kvinge
8c039c9c8b Release 0.6.5 2019-09-30 22:05:42 +02:00
Jonas Kvinge
c1d9bc046d Change context default 2019-09-30 22:05:03 +02:00
Jonas Kvinge
cd99fac7ed Fix OSD Pretty upper left (0,0) position and positioning on Windows
Fixes #269
2019-09-30 20:32:34 +02:00
Jonas Kvinge
f4489e6807 No need to initialize SimpleMetaBundle here 2019-09-30 20:31:18 +02:00
Jonas Kvinge
f2078271b6 Only update scrobble point in SetStreamMetadata when length is changed 2019-09-30 18:58:55 +02:00
Jonas Kvinge
a3ae9acebb Listenbrainz: don't send "various artists" as artist 2019-09-29 13:50:24 +02:00
Jonas Kvinge
b0580265ca Listenbrainz: don't send "various artists" as album artist
Fixes #273
2019-09-29 13:31:46 +02:00
Jonas Kvinge
b4f012392a Fix full validation of appdata file
Fixes #271
2019-09-29 13:29:52 +02:00
Jonas Kvinge
1598809f55 Fix OSD reposition image 2019-09-25 19:01:55 +02:00
Jonas Kvinge
6d4f7aa61f Turn back git revision 2019-09-25 18:54:22 +02:00
Jonas Kvinge
382080ae4e Release 0.6.4 2019-09-25 13:54:16 +02:00
Strawbs Bot
66ac529bca Update translations 2019-09-25 12:01:08 +02:00
Jonas Kvinge
ab72207027 Change domain 2019-09-24 00:06:37 +02:00
Jonas Kvinge
e3aebf1ca2 Update maketarball.sh.in 2019-09-23 22:23:30 +02:00
Jonas Kvinge
0e5820c038 Update file types 2019-09-23 21:52:48 +02:00
Jonas Kvinge
780e4d34df Update Changelog 2019-09-23 21:32:56 +02:00
Jonas Kvinge
a8700572b7 Allow empty artist and album in Subsonic song replies
Fixes #266
2019-09-23 19:20:11 +02:00
Jonas Kvinge
055516312a Fix signature in Qobuz stream url request
Fixes #267
2019-09-23 19:18:12 +02:00
Jonas Kvinge
76b4a6585e Update copyrights 2019-09-23 19:17:41 +02:00
Jonas Kvinge
b57535c5ad Make gstreamer discoverer handle next url too 2019-09-23 01:03:03 +02:00
Jonas Kvinge
eb10a15eee Reset next item temporary metadata 2019-09-23 01:00:55 +02:00
Jonas Kvinge
30ed362a8c Increase preload gap 2019-09-23 01:00:27 +02:00
Jonas Kvinge
34f071ed1d Change conflicting variable 2019-09-23 00:34:29 +02:00
Jonas Kvinge
3d3d641e1c Fix player not using preloaded stream url breaking gapless playback
Fixes #26
2019-09-22 22:47:07 +02:00
Jonas Kvinge
10c11f2c3d Fix compile on older Qt than 5.11 2019-09-22 22:43:54 +02:00
Jonas Kvinge
cec2745dc0 Qobuz: Send mac address as device_manufacturer_id in login 2019-09-22 17:10:04 +02:00
Jonas Kvinge
8d15edb063 Add function to get mac address 2019-09-22 17:06:45 +02:00
Jonas Kvinge
38cf3dc141 Only set user agent if it's missing in NetworkAccessManager 2019-09-22 17:05:26 +02:00
Jonas Kvinge
c146290e07 Add function to get mac address 2019-09-22 17:04:57 +02:00
Jonas Kvinge
a1745289be Correct app id header 2019-09-21 18:54:15 +02:00
Jonas Kvinge
5a7dbfc87a qobuz: add headers to requests 2019-09-21 18:53:12 +02:00
Jonas Kvinge
f645950a8f Change all API urls to https 2019-09-20 23:22:27 +02:00
Jonas Kvinge
b273da0b5e Switch to openssl 1.1 in nsi 2019-09-20 18:49:53 +02:00
Jonas Kvinge
689f1eb0c5 Add libgiognutls.dll to nsi 2019-09-20 18:31:35 +02:00
Jonas Kvinge
b0a2edf5fa tagreader: remove use of deprecated taglib functions 2019-09-20 18:07:43 +02:00
Jonas Kvinge
e7b5e02657 Update taglib 2019-09-20 18:07:13 +02:00
Jonas Kvinge
2d4efd6cf0 Add libbrotlicommon.dll 2019-09-19 20:56:39 +02:00
Jonas Kvinge
defc0ada78 Fix compile warnings 2019-09-19 17:44:14 +02:00
Jonas Kvinge
2f3f4f609c Rename libbrotlidec-0.dll to libbrotlidec.dll 2019-09-19 17:24:49 +02:00
Jonas Kvinge
4a9c9f8cd4 gstreamer: disconnect callbacks, avoid gst_discoverer_stop 2019-09-17 22:42:51 +02:00
Jonas Kvinge
bdc089290d Add fallthrough comments and remove -Wimplicit-fallthrough=0
Signed-off-by: Jonas Kvinge <jonas@jkvinge.net>
2019-09-16 21:20:12 +02:00
Strawbs Bot
cf9f48d8da Update translations 2019-09-16 20:57:06 +02:00
Jonas Kvinge
2d67279180 Fix minor code issues 2019-09-15 20:27:32 +02:00
Jonas Kvinge
83e10aac27 Fix exiting macos devicelister 2019-09-15 01:12:05 +02:00
Jonas Kvinge
9972124aa1 Add gstreamer plugins to macdeploy.py 2019-09-14 20:30:06 +02:00
Jonas Kvinge
d0eb1ba96e Avoid gst_discoverer_stop on all OSes except Linux 2019-09-14 20:11:29 +02:00
Jonas Kvinge
f5d2910638 Add libbrotlidec-0.dll and libpsl-5.dll to nsi 2019-09-12 20:38:13 +02:00
Jonas Kvinge
1cafaf3a79 Disable video for playbin 2019-09-12 18:07:10 +02:00
Jonas Kvinge
fccc133cb2 Update gstreamer plugins in nsi 2019-09-11 21:00:44 +02:00
Jonas Kvinge
9fb50b8a73 Update taglib 2019-09-11 00:36:03 +02:00
Jonas Kvinge
c66c1e17d3 Make gstreamer pipeline detect filetype 2019-09-09 22:11:13 +02:00
Jonas Kvinge
4496760340 Dont search for lyrics if tags are empty 2019-09-09 21:48:22 +02:00
Jonas Kvinge
506200d2ee Update supported audio in README 2019-09-09 21:47:54 +02:00
Jonas Kvinge
04898a3b01 Add Ogg Vorbis test audio file 2019-09-09 21:47:39 +02:00
Jonas Kvinge
f67fb53308 Fix conflicting variable 2019-09-09 18:17:56 +02:00
Jonas Kvinge
c5b78dde04 Convert remaining foreach to C++11 for loops 2019-09-09 00:53:05 +02:00
Jonas Kvinge
57d9c87de6 Replace NULL with nullptr 2019-09-08 21:18:26 +02:00
Jonas Kvinge
795f95d855 Add gstreamer stream discoverer workaround for Windows
- gst_discoverer_stop seem to block
2019-09-08 21:07:56 +02:00
Jonas Kvinge
c0ebbc8e2f Fix ampache compatibility 2019-09-08 19:46:51 +02:00
Jonas Kvinge
fb377c32ea Initialize filetype and bitrate 2019-09-08 17:36:50 +02:00
Jonas Kvinge
e13faff2d7 Add missing include for assert 2019-09-08 00:36:32 +02:00
Jonas Kvinge
b0c5348116 Remove comment 2019-09-07 23:50:36 +02:00
Jonas Kvinge
b462ec022a Remove unused variable 2019-09-07 23:50:26 +02:00
Jonas Kvinge
e45a0bf24b Add stream discoverer to gstreamer pipeline and continuous updating of bitrate 2019-09-07 23:34:13 +02:00
Jonas Kvinge
8962644ba8 Improvements to device manager
- Mount and unmount devices in lister thread
- Safely close watcher and backends for devices
- Enable abort loading device
- Fix MTP connection
2019-09-07 23:30:35 +02:00
Jonas Kvinge
ad5e04bbcc Always ignore closeEvent 2019-09-05 19:07:58 +02:00
Jonas Kvinge
6fc7baec39 Add Russian 2019-09-01 23:59:03 +02:00
Jonas Kvinge
acd49a9c40 Update translations 2019-09-01 18:00:04 +02:00
Jonas Kvinge
bf04358a47 Replace media buttons 2019-08-29 22:11:22 +02:00
Jonas Kvinge
b7e13c7b86 Fix windows thumbbar 2019-08-29 21:32:52 +02:00
Jonas Kvinge
9b45f0661e Add option to bundle gstreamer plugins on Linux 2019-08-27 22:29:48 +02:00
Jonas Kvinge
2a61b1202b Set subsonic API version to 1.13.0 2019-08-27 17:26:49 +02:00
Jonas Kvinge
114f862ea4 TagLib MP4 - setTrack()/setYear() accepts 0 to remove the tag 2019-08-26 23:50:14 +02:00
Jonas Kvinge
5a490a8b5a SingleApplication: Fallback to qgetenv for username on Windows too 2019-08-26 17:31:49 +02:00
Jonas Kvinge
8fdc1c1b21 SingleApplication: Use initialization list 2019-08-26 17:22:11 +02:00
Jonas Kvinge
32cac90720 SingleApplication: Remove debug output of username 2019-08-26 02:18:02 +02:00
Jonas Kvinge
b2160255d3 SingleApplication: Use geteuid / getpwuid to get username and reformat
sources
2019-08-26 02:00:47 +02:00
Jonas Kvinge
7fe1f4de93 Fix compile on OpenBSD 2019-08-26 01:37:22 +02:00
Jonas Kvinge
1fbb1b1524 Keep original window size when restoring from system tray 2019-08-23 20:43:58 +02:00
Jonas Kvinge
0105a53765 Use qgetenv instead of QProcess/whoami to get username 2019-08-22 21:38:54 +02:00
Jonas Kvinge
bd5ab80276 Use FollowRedirectsAttribute everywhere 2019-08-22 19:28:54 +02:00
Jonas Kvinge
387d790228 Fix spelling in contributor variables in about 2019-08-22 18:49:03 +02:00
Jonas Kvinge
d199a2be0d Fix typos and spelling 2019-08-22 18:45:32 +02:00
Jonas Kvinge
f81ecffda6 Don't treat songs with different album as duplicates 2019-08-21 20:46:08 +02:00
Jonas Kvinge
882f80de1e Add FollowRedirectsAttribute 2019-08-21 00:01:38 +02:00
Jonas Kvinge
4359f2a0ce Follow redirects in subsonic 2019-08-20 23:32:20 +02:00
Jonas Kvinge
9c485c4d94 Split NetworkTimeouts and RedirectFollower to it's own files
- Follow redirects by default
2019-08-20 23:31:23 +02:00
Jonas Kvinge
1330197036 Small subsonic changes
- Follow redirect in subsonic ping test
- Reset networkaccessmanager to make verify certificate setting take
affect immediately
2019-08-20 12:33:01 +02:00
Jonas Kvinge
0b0d5fa227 Add back libgstlame in macdeploy 2019-08-18 22:08:32 +02:00
Jonas Kvinge
45424b8a52 Change CFBundleIdentifier 2019-08-18 01:00:47 +02:00
Jonas Kvinge
0956acbdfb Add missing comma 2019-08-17 01:04:48 +02:00
Jonas Kvinge
654f553d8f Move InformOfCurrentSongChange() to fix crash when active playlist item
is reset
2019-08-17 00:52:29 +02:00
Jonas Kvinge
e4b6e20db6 Turn playlist glow effect off by default on macOS 2019-08-17 00:00:12 +02:00
Jonas Kvinge
b4f6c6f869 Include NSWindow in mac_startup.mm 2019-08-16 22:52:50 +02:00
Jonas Kvinge
16edc52bae Fix deprecated macOS key modifiers 2019-08-16 22:27:48 +02:00
Jonas Kvinge
a33e6c03e4 Use QPixmap::toImage() and QImage.toCGImage() instead 2019-08-16 22:26:30 +02:00
Jonas Kvinge
4dccb40bf9 Add missing uninstall for xine plugins in nsi 2019-08-14 23:29:44 +02:00
Jonas Kvinge
9ce1981f93 Fix copyright in iconloader 2019-08-14 23:28:55 +02:00
Jonas Kvinge
6374c77aa8 Read Json error when possible 2019-08-12 22:24:05 +02:00
Jonas Kvinge
871bb391d6 Add lyrics from lololyrics.com 2019-08-12 22:06:01 +02:00
Jonas Kvinge
c3903a7b35 Add lyrics from lyrics.ovh 2019-08-12 18:11:01 +02:00
Jonas Kvinge
6aabf82c11 Update issue templates 2019-08-11 23:33:15 +02:00
Jonas Kvinge
89b624aadf Update issue templates 2019-08-11 23:27:46 +02:00
Jonas Kvinge
67317292ee Create FUNDING.yml 2019-08-11 23:11:26 +02:00
Jonas Kvinge
42e7f64856 Make marking songs unavailable optional 2019-08-11 22:30:28 +02:00
Jonas Kvinge
79d963fd65 Change variable name 2019-08-11 22:29:02 +02:00
Jonas Kvinge
44649fd950 Remove remaining references to SPMediaKeyTap 2019-08-11 21:06:25 +02:00
Martin Delille
0e97f99f93 remove 3rdparty SPMediaKeyTap (#239) 2019-08-11 20:44:32 +02:00
Jonas Kvinge
4f52ceb3e0 Make fancy tabbar background color configurable 2019-08-08 23:16:45 +02:00
Jonas Kvinge
61253b5551 Update taglib 2019-08-07 19:29:40 +02:00
llucps
12bd6f0d9b Enable About Strawberry and QT menu on macOS (#237)
* Enable About Strawberry and QT menu on macOS

* Enable About Strawberry and QT menu on macOS
2019-08-07 18:26:11 +02:00
SamTShaw
a32010e03b Ipod Playlist Support (#220)
* Ipod Playlist Support

Copy a whole playlist to the ipod and create an entry in Playlists on
the iPod

* Fix formatting and indentation

Fix indenting and formatting to be consistent
2019-08-07 17:13:40 +02:00
Jonas Kvinge
4a934c9dab Remove use of some deprecated code and cleanup other macOS code 2019-08-06 20:31:54 +02:00
Jonas Kvinge
20766c1feb Fix systemtray icon on macOS 2019-08-06 20:31:31 +02:00
Jonas Kvinge
6cd4de548f Turn back git revision 2019-08-06 15:17:31 +02:00
Jonas Kvinge
8ac4dea8f1 Release 0.6.3 2019-08-05 23:26:19 +02:00
Jonas Kvinge
e8af3e8d3c Simplify if statement 2019-08-05 23:03:40 +02:00
Jonas Kvinge
cd05b10168 Remove extra debugging 2019-08-05 23:03:04 +02:00
Jonas Kvinge
140935bd8c Add album - disc grouping 2019-08-05 19:17:31 +02:00
Jonas Kvinge
ecb122d93c Make collection watcher unwatch removed directories
Fixes #233
2019-08-05 18:40:47 +02:00
Jonas Kvinge
c6e08e0039 Fix crash in internet services 2019-08-05 18:38:27 +02:00
Jonas Kvinge
be1e14df81 Fix musicbrainz tagfetcher 2019-08-04 16:02:22 +02:00
Jonas Kvinge
28bca261fb Fix mageia rpm suffix 2019-08-04 03:16:30 +02:00
Jonas Kvinge
2ab5bc1ad2 Turn back git revision 2019-08-04 03:16:08 +02:00
Jonas Kvinge
10c57cf6da Release 0.6.2 2019-08-03 13:57:54 +02:00
Jonas Kvinge
e65c5eeab1 Release 0.6.1 2019-08-03 13:29:22 +02:00
Jonas Kvinge
85fad35dc2 Update README.md 2019-08-03 13:15:43 +02:00
Jonas Kvinge
c992768efe Remove ChartLyrics (service is down) 2019-08-03 13:08:14 +02:00
Jonas Kvinge
7b13c0d059 Fix watching new subdirs in collection watcher 2019-08-02 23:58:47 +02:00
Jonas Kvinge
2d3f41da6f Automatically remove devices with old schemas when upgrading 2019-08-02 22:58:30 +02:00
Jonas Kvinge
43b9941dc8 Make sqlite3 fts5 check fatal 2019-08-02 21:12:18 +02:00
Jonas Kvinge
dcf27f54aa Add rpm syntax for mageia 2019-08-02 20:25:46 +02:00
Jonas Kvinge
fbf6b69e75 Update Changelog 2019-08-01 22:01:27 +02:00
Jonas Kvinge
0bfebd90da Add CMAKE_REQUIRED_FLAGS --std=c++11 to check_cxx_source_runs 2019-08-01 21:49:19 +02:00
Jonas Kvinge
e7c3dafa36 Handle a case where the playing widget is gets stuck when switch fast
between context and other widgets
2019-08-01 21:15:46 +02:00
Jonas Kvinge
e90a36da79 Fix track slider popup being stuck 2019-08-01 20:03:37 +02:00
Jonas Kvinge
93f33615ad Don't use check_cxx_source_runs when cross-compiling 2019-07-31 23:06:59 +02:00
Jonas Kvinge
99569081c9 Simply some song checks and make url always unique by using stream url
instead
2019-07-31 22:26:51 +02:00
Jonas Kvinge
588a0b3c41 Add cmake test for sqlite3 FTS5 2019-07-31 20:05:38 +02:00
Jonas Kvinge
4a1118ceb3 Fix macOS build 2019-07-31 20:03:24 +02:00
Jonas Kvinge
a65415bc10 Fix device schema 2019-07-30 22:56:33 +02:00
Jonas Kvinge
8a0e66bf11 Switch to FTS5 with unicode61 (#229)
* Switch to FTS5 with unicode61

* Update required sqlite version in README

* Update README

* Change back db file
2019-07-30 22:45:22 +02:00
Jonas Kvinge
02cda47c28 Disable trackslider popup on macos 2019-07-30 22:11:09 +02:00
Jonas Kvinge
d34a323a81 Handle cases where playlist background album gets stuck on error 2019-07-30 21:32:56 +02:00
Jonas Kvinge
4166ae8db0 Check HttpStatusCodeAttribute 2019-07-30 21:32:20 +02:00
Jonas Kvinge
955b906b52 Add Song() != operator 2019-07-30 21:31:26 +02:00
Jonas Kvinge
0d424aa81e Dont fetch lyrics again if only url is changed through url handler 2019-07-30 21:30:52 +02:00
Jonas Kvinge
80acbfa56a Update contributors 2019-07-30 21:30:08 +02:00
Jonas Kvinge
a7cac24004 Unref pipeline in moodbar on failure 2019-07-28 23:42:12 +02:00
Jonas Kvinge
2927bcf09d Merge branch 'master' of github.com:jonaski/strawberry 2019-07-28 14:58:52 +02:00
Jonas Kvinge
08dee6bb4f Fix error message from url handler 2019-07-28 14:58:34 +02:00
Jonas Kvinge
08c92f906a Update translations 2019-07-27 13:25:51 +02:00
Jonas Kvinge
8e9c9802d1 Change new line to unix 2019-07-27 12:34:27 +02:00
Jonas Kvinge
138df66d5e Use common nsi file 2019-07-27 12:32:28 +02:00
Jonas Kvinge
0a71347e9a Fix playlist shortcuts 2019-07-26 22:46:04 +02:00
Jonas Kvinge
2a541a7b9c Update README 2019-07-26 20:55:20 +02:00
Jonas Kvinge
705c12e8da Update README and Changelog 2019-07-26 20:52:47 +02:00
Jonas Kvinge
cf33df1339 Log system type on startup 2019-07-26 19:24:16 +02:00
Jonas Kvinge
87e543b5ef Fix collection query 2019-07-26 19:14:15 +02:00
Jonas Kvinge
81caec99b7 Fix closing databases 2019-07-25 17:56:28 +02:00
Jonas Kvinge
41484f8673 Fix exit 2019-07-24 23:29:09 +02:00
Jonas Kvinge
da0d61f36a Fix regression in playlist backend caused by previous commits 2019-07-24 21:37:09 +02:00
Jonas Kvinge
6f3bc74db0 Fix cross-compile for windows 2019-07-24 19:34:33 +02:00
Jonas Kvinge
af3bd6ec2f Use QNetworkAccessManager::supportedSchemes() 2019-07-24 19:27:00 +02:00
Jonas Kvinge
b5eb13449b Safely close database connections and delete backends
Also fix NewClosure leak caused by disconnected object signals
2019-07-24 19:16:51 +02:00
Jonas Kvinge
bd78e8c275 Fix memory leaks 2019-07-22 20:53:05 +02:00
Jonas Kvinge
2df21081a1 Fix pixmap cache in collection model
- Properly remove both from pixmap cache, pending art and pending cache keys when songs are deleted
- Increase default pixmap cache
- Clear pixmap cache when model is reset
2019-07-22 19:57:53 +02:00
Jonas Kvinge
ffebff4ea9 Fix uninitialized variable 2019-07-21 21:28:45 +02:00
Jonas Kvinge
2885bc99ca Fix memory leak in Database::BackupFile 2019-07-21 18:50:16 +02:00
Jonas Kvinge
2657b80adb Fix memory leak in tagreader 2019-07-21 15:12:28 +02:00
Jonas Kvinge
ebfc106701 Update libcdio 2019-07-21 13:19:27 +02:00
Jonas Kvinge
4d60ca951d Update Changelog 2019-07-21 00:05:54 +02:00
Jonas Kvinge
acf5c57599 Use InitArtManual in InitFromFilePartial 2019-07-20 22:53:01 +02:00
Jonas Kvinge
927df5ff39 Bump LSMinimumSystemVersion to 10.13.4 2019-07-20 15:26:36 +02:00
Jonas Kvinge
31fc031267 Fix missing qt plugins in macOS travis-ci build (#225) 2019-07-20 15:23:22 +02:00
Jonas Kvinge
02794e0ebd Add libzstd and libtasn1-6 2019-07-20 02:26:44 +02:00
Jonas Kvinge
ea6cce7068 Fix mtp device support 2019-07-19 19:56:37 +02:00
SamTShaw
e4cefeaa8f Ipod Support when listed from Udisks2 (#219)
Udisks2 didn't check to see if the device was actually an ipod, like
GIODeviceLister does.
2019-07-19 19:22:14 +02:00
Jonas Kvinge
d12d5fd8ae Update .travis.yml 2019-07-18 23:55:30 +02:00
Jonas Kvinge
0617c3fdc0 Update .travis.yml 2019-07-18 23:09:01 +02:00
Jonas Kvinge
f00eedf57e Update .travis.yml 2019-07-18 22:12:48 +02:00
Jonas Kvinge
6d36ae6197 Update .travis.yml 2019-07-18 21:35:07 +02:00
Martin Delille
45125abb8f Fix Finder icon (#216)
- Regenerate new strawberry.icns
- Fix Info.plist CFBundleIconFile
- Install Info.plist to strawberry.app
2019-07-17 23:13:15 +02:00
Jonas Kvinge
045b0cd075 Remove bogus svg 2019-07-17 22:35:19 +02:00
Martin Delille
ff3333e1bf Use create-dmg to generate MacOS release (#215) 2019-07-17 22:05:41 +02:00
Martin Delille
7663c5e149 MacOS CI improvement (#214)
* travis ci osx: simpler lib path

* Remove unnecessary sudo
2019-07-17 21:10:17 +02:00
Jonas Kvinge
4d4748a0a8 Update protobuf 2019-07-14 12:26:07 +02:00
Jonas Kvinge
f8f84ed09e Only replace non-ascii characters when not allowing extended ascii 2019-07-14 03:16:53 +02:00
Jonas Kvinge
e7de7ebbfa Use iconv to replace non-ascii characters 2019-07-14 03:08:19 +02:00
Jonas Kvinge
c9f01f4bc4 Remove old devicekit udisks backend 2019-07-13 23:00:25 +02:00
Jonas Kvinge
f2675adc05 Remove vlc, xine and phonon from deb 2019-07-13 22:52:05 +02:00
Jonas Kvinge
75beaa3684 Make sure device is valid before adding it 2019-07-13 18:28:29 +02:00
Jonas Kvinge
079495cc32 Remove usless include in cmake 2019-07-13 18:02:25 +02:00
Jonas Kvinge
c8bd89e56f Remove support for old imobiledevice uuid, it won't work anyway because
of other changes
2019-07-13 17:55:50 +02:00
Jonas Kvinge
b089ba2e04 Free memory of service descriptor and fix possible crashes 2019-07-13 17:33:58 +02:00
Jonas Kvinge
adbf8495c6 Add libgstaiff.so to macdeploy 2019-07-12 20:32:52 +02:00
Jonas Kvinge
ad062862d8 Add device url to mtp loader failure message 2019-07-12 20:32:30 +02:00
Gavin D. Howard
3a86a93154 Extend article sorting to artist and album artist (#210) 2019-07-11 16:51:25 +02:00
Jonas Kvinge
dcf0d6f72d Fix thread crash in mtpdevice 2019-07-11 00:28:19 +02:00
Jonas Kvinge
83d3725240 Fix MessageReply crash in tagreader 2019-07-10 22:40:30 +02:00
Jonas Kvinge
ce1dd69557 Change defaults in device schema 2019-07-10 20:04:05 +02:00
Jonas Kvinge
4081bdd752 Fix ctime in tagreader 2019-07-10 20:02:51 +02:00
Jonas Kvinge
c3f1e312b3 Change defaults in schemas 2019-07-10 20:02:21 +02:00
Jonas Kvinge
d820878b89 Fix oauth checkbox in tidal settings 2019-07-09 22:53:50 +02:00
Jonas Kvinge
7fa1461d5e Use QUrl::isLocalFile() 2019-07-09 21:43:56 +02:00
Jonas Kvinge
f4b1ef4d04 Set initial position of OSDPretty 2019-07-09 20:28:43 +02:00
Jonas Kvinge
656130a739 Replace QDesktopWidget in OSDPretty 2019-07-09 19:49:15 +02:00
Jonas Kvinge
aa8679dff5 Fix TryLoadPixmap and ShowCover 2019-07-09 01:14:58 +02:00
Jonas Kvinge
f94a3095fd Fix TryLoadPixmap and ShowCover 2019-07-09 01:05:42 +02:00
Jonas Kvinge
e11958dd58 Fix index in InternetSearchModel 2019-07-09 01:02:54 +02:00
Jonas Kvinge
a5a251a964 Remove use of QDesktopWidget in QSearchField 2019-07-09 00:05:08 +02:00
Jonas Kvinge
8cb1015a35 Remove use of QDesktopWidget in settingsdialog 2019-07-09 00:04:52 +02:00
Jonas Kvinge
b5dd90b2d5 Check QT_VERSION_CHECK for QImage::sizeInBytes() or QImage::byteCount() 2019-07-08 23:35:43 +02:00
Jonas Kvinge
5f7efee00e ifdef QFontMetrics::horizontalAdvance to make it work with older Qt 2019-07-08 23:27:45 +02:00
Jonas Kvinge
4150e3efde Use QT_VERSION to check whether to use QFileInfo::birthTime() or
QFileInfo::created()
2019-07-08 22:54:13 +02:00
Jonas Kvinge
8ebcb71e6e Replace all uses of QSignalMapper with lambda expressions 2019-07-08 22:27:45 +02:00
Jonas Kvinge
f3e852c042 Replace QImage::byteCount() with sizeInBytes() 2019-07-08 22:25:56 +02:00
Jonas Kvinge
5e2a07d144 Remove unused typedef 2019-07-08 22:24:47 +02:00
Jonas Kvinge
25f6231e9d Replace QString::null with QString() 2019-07-08 22:24:00 +02:00
Jonas Kvinge
beeba88ea5 Replace QModelIndex child() with index() 2019-07-08 22:23:15 +02:00
Jonas Kvinge
cc3d454d60 Add lyrics to edit tag dialog 2019-07-08 22:21:12 +02:00
Jonas Kvinge
afb583cff4 Remove unused typedef 2019-07-08 22:20:41 +02:00
Jonas Kvinge
561fa66393 Replace QFontMetrics::width with horizontalAdvance 2019-07-08 22:20:02 +02:00
Jonas Kvinge
870dc0d36f Replace QFontMetrics::width with horizontalAdvance, dark with darker, background() with window() and QString::null with QString() 2019-07-08 22:19:14 +02:00
Jonas Kvinge
51462dee1e Use QUrl::fromEncoded 2019-07-08 22:10:43 +02:00
Jonas Kvinge
c752d28c6a Change QFileInfo::created() to birthTime(), read lyrics from MP3 files 2019-07-08 22:01:34 +02:00
Jonas Kvinge
d5ca0ca283 Change compiler flags 2019-07-08 21:59:07 +02:00
Jonas Kvinge
f371b3a338 Fix tidal request 2019-07-08 17:08:10 +02:00
Jonas Kvinge
93fc4a2c86 Fix Tidal api URL 2019-07-08 08:10:52 +02:00
Jonas Kvinge
f10d3f86cc Fix Mpris2 AlbumCoverLoaded 2019-07-08 00:13:45 +02:00
Jonas Kvinge
af17b0015b Fix InitArtManual() to use album artist 2019-07-07 21:56:30 +02:00
Jonas Kvinge
61af1d1c72 Update file filter 2019-07-07 21:36:05 +02:00
Jonas Kvinge
65780e1672 Improve album cover searching and cover manager, use HttpStatusCodeAttribute and QSslError for services
- Improve album cover manager
- Change art_automatic and art_manual to QUrl
- Refresh collection album covers when new album covers are fetched
- Fix automatic album cover searching for local files outside of the collection
- Make all Json services check HttpStatusCodeAttribute
- Show detailed SSL errors for Subsonic, Tidal and Qobuz
2019-07-07 21:14:24 +02:00
Jonas Kvinge
c92a7967ea Change filename to url 2019-07-06 14:54:41 +02:00
Jonas Kvinge
60aed593b3 Fix download albumcovers setting 2019-07-06 00:16:13 +02:00
Jonas Kvinge
044f347729 Re-enable tidal oauth settings 2019-07-06 00:02:25 +02:00
Jonas Kvinge
aec9df1882 Switch to queue2 for probe queue (#204) 2019-07-05 23:50:57 +02:00
Jonas Kvinge
4f0a2515f8 Tweak the background image settings a bit
- Disable "do not cut" option when keep aspect ratio is unchecked
- Shorten text and add spacer to widget
2019-07-05 01:07:19 +02:00
Gavin D. Howard
5cde33711e Make playlist ignore articles when sorting (#202)
This is more correct, and the other way is driving me crazy.
2019-07-05 00:13:51 +02:00
Jonas Kvinge
64750025f6 Update libnettle 2019-07-04 22:51:17 +02:00
Jonas Kvinge
cd1cbfdffe Update libhogweed 2019-07-04 22:44:42 +02:00
Jonas Kvinge
d413f2c3a7 Revert "Try queue2 for probe queue"
This reverts commit a9c162476c.
2019-07-03 02:24:32 +02:00
Jonas Kvinge
ba314dd734 Set source in backends to fix losing source in InitFromFilePartial()
when updating path
2019-07-02 00:48:40 +02:00
Jonas Kvinge
9105b7615c Change to reference 2019-07-02 00:48:09 +02:00
Jonas Kvinge
a9c162476c Try queue2 for probe queue 2019-07-02 00:47:10 +02:00
Jonas Kvinge
732b37aca0 Use QMap::contains() 2019-07-01 19:20:57 +02:00
Jonas Kvinge
291b5fad7f Updated Changelog 2019-07-01 01:08:07 +02:00
Jonas Kvinge
c2a6def8b9 Rename cache album covers to download album covers and only do it for
favorite requests
2019-07-01 01:01:30 +02:00
Jonas Kvinge
9083c578cc Add live scanning (#199) 2019-06-30 21:06:07 +02:00
m4x3t
bcfd1d39bb Modify stretch background image functionality (#198)
* Modify stretch background image functionality

Changes the stretch functionality to fill out
the playlist background completely by zooming
the image (if keep aspect ratio is selected)
instead of filling it only height-wise.

* Add option to keep old background fill

* Fix playlist background image width and height

* Fix width calculation

* Remove old calculations
2019-06-30 19:37:05 +02:00
Jonas Kvinge
1185b910c4 Ops 2019-06-29 20:29:37 +02:00
Jonas Kvinge
540b6eba04 Fix hardcoded systemtray icon 2019-06-29 20:18:54 +02:00
Jonas Kvinge
47f4287e64 Merge remote-tracking branch 'origin/musicbrainz' 2019-06-29 19:58:10 +02:00
Jonas Kvinge
264c6e259b Respect rate limiting when fetching tags from musicbrainz 2019-06-29 19:57:20 +02:00
m4x3t
0bbe9838c4 Add option: notification on playback resume (#196)
* Add option: notification on playback resume

This adds an optional setting to show the
notification that is displayed when changing
the track when resuming playback as well.

* Modify resume notification calling

This adds a new signal "Resumed" that is emitted
when the player status is changed from Paused
to Playing.
The AlbumArtLoaded function will only be called
again when playback is manually resumed, and not
when the player is started for the first time
or when the track is changed.
2019-06-29 19:54:27 +02:00
Jonas Kvinge
2fe337eac3 Partial revert travis_terminate 2019-06-29 19:18:26 +02:00
Jonas Kvinge
5d7caafdf7 Terminate if one of the commands fail inside an if statement 2019-06-29 16:31:47 +02:00
Jonas Kvinge
f416ef925b Only create private key if DEPLOY_KEY_ENC is set 2019-06-29 15:57:30 +02:00
Jonas Kvinge
60bd90848b Add tests (#193) 2019-06-28 01:33:22 +02:00
Jonas Kvinge
b9f4407815 Change to const QString& 2019-06-23 18:34:03 +02:00
Jonas Kvinge
f9cd2639ff Remove annoying <year> Remaster from album and song titles 2019-06-23 18:32:15 +02:00
Jonas Kvinge
aeb36e8665 Simplify generating queries in lyrics providers 2019-06-23 16:45:00 +02:00
Jonas Kvinge
057482a3e5 Lower default search limits 2019-06-23 00:20:53 +02:00
Jonas Kvinge
15721da46e Read duration, samplerate and bit depth from stream url replies 2019-06-22 08:39:30 +02:00
Jonas Kvinge
f12b82b5ce Move checkboxes into preferences groupbox in Subsonic settings 2019-06-22 08:38:50 +02:00
Jonas Kvinge
10dc725942 Attempt to fix MessageReply crash when saving tags 2019-06-22 08:36:02 +02:00
Jonas Kvinge
5bd5cdf435 Make sure album_id is not null 2019-06-22 08:33:16 +02:00
Jonas Kvinge
84b3603c08 Fix initialization 2019-06-21 02:26:37 +02:00
Jonas Kvinge
8f59b96731 Update Changelog 2019-06-21 00:12:07 +02:00
Jonas Kvinge
e1de110dd7 Validate configuration better 2019-06-20 20:49:02 +02:00
Jonas Kvinge
8b6fd3d594 Make text selectable 2019-06-20 18:46:14 +02:00
Jonas Kvinge
d7761e8d79 Change parameters to const 2019-06-20 17:09:07 +02:00
Jonas Kvinge
751d652a2f Change title 2019-06-20 17:02:29 +02:00
Jonas Kvinge
647e7e708a Add confirmation box for opening songs in file browser 2019-06-20 17:00:10 +02:00
Jonas Kvinge
505c0eeae2 Set Unknown error if error is missing in Subsonic too 2019-06-20 16:37:37 +02:00
Jonas Kvinge
bf4001968e Change to const references, make search progress and status pass search id 2019-06-20 16:33:28 +02:00
Jonas Kvinge
9d222e2c57 Rename filename to url, change album_id to string and recreate songs tables (#182) 2019-06-20 02:14:44 +02:00
Jonas Kvinge
033300d659 Disable remove favorites from context menu in internetsongsview 2019-06-19 23:26:15 +02:00
Jonas Kvinge
4f2b04bd8f Enable login buttons when login attempt is complete 2019-06-19 23:15:15 +02:00
Jonas Kvinge
f8b9bb4b0f Remove user id and country code setting 2019-06-19 22:20:21 +02:00
Jonas Kvinge
8ce8e320c3 Change ids to qint64 2019-06-19 20:40:11 +02:00
Jonas Kvinge
7c0ab4212b Fix minor code issues with Qobuz 2019-06-19 12:57:12 +02:00
Jonas Kvinge
26633e0982 Fix compile error 2019-06-19 02:39:11 +02:00
Jonas Kvinge
d27a3f1f33 Set Subsonic API version to 1.15.0 2019-06-19 02:30:28 +02:00
Jonas Kvinge
89252d0dba Add Qobuz support (#181) 2019-06-19 02:22:11 +02:00
Jonas Kvinge
dbd2edf442 Fix minor code issues in tidal 2019-06-18 23:39:16 +02:00
Jonas Kvinge
dabd6f8284 Fix Subsonic API path 2019-06-18 23:25:10 +02:00
Jonas Kvinge
beb3ab463f Add missing subsonic fts songs table 2019-06-18 14:41:48 +02:00
Jonas Kvinge
1d67b623e0 Use single url setting instead for subsonic, check http code 2019-06-18 01:22:03 +02:00
Jonas Kvinge
2c8cde4d91 Add scheme setting for subsonic 2019-06-18 00:28:36 +02:00
Jonas Kvinge
8cc1d48115 Fix max concurrent requests in tidal 2019-06-18 00:02:43 +02:00
Jonas Kvinge
ec2468fb8d Remove unneeded include 2019-06-18 00:02:22 +02:00
Jonas Kvinge
7b54cef23b Add Subsonic support (#180) 2019-06-17 23:54:24 +02:00
Jonas Kvinge
a9da8811fc Turn off ssl-strict in gst pipeline 2019-06-17 23:01:38 +02:00
Jonas Kvinge
60e86b9881 Fix fading options 2019-06-17 02:23:51 +02:00
Jonas Kvinge
a6766f3c99 Fix gapless playback when using url handler 2019-06-15 19:32:26 +02:00
Jonas Kvinge
e59c3c6f70 Add MediaUrl function for playlistitems 2019-06-15 19:31:43 +02:00
Jonas Kvinge
49aa344d8b Change index to idx 2019-06-15 19:31:16 +02:00
Jonas Kvinge
4fed6ab298 Hide tidal oauth settings until streaming is possible 2019-06-14 23:02:57 +02:00
Jonas Kvinge
58eab4d3bc Fix verifying xml in tidal stream url 2019-06-14 23:02:33 +02:00
Jonas Kvinge
5ef5da687d Fix macOS build 2019-06-12 06:34:59 +02:00
Jonas Kvinge
4875d319dc Add love button 2019-06-12 00:38:52 +02:00
Jonas Kvinge
fb5a53435e Remove unused function 2019-06-11 19:28:48 +02:00
Jonas Kvinge
4f805d65b3 Fix need_login function 2019-06-11 19:26:00 +02:00
Jonas Kvinge
56ffb0deb1 Fix code challenge in tidal oauth 2019-06-10 02:29:57 +02:00
Jonas Kvinge
c0c1457073 Add optional oauth authentication for tidal 2019-06-09 19:29:25 +02:00
Jonas Kvinge
85a0748ad9 Add x-scheme-handler/tidal to desktop file 2019-06-09 03:10:37 +02:00
Jonas Kvinge
c1939a36dd Change CMakeLists.txt to use list APPEND for Qt5 2019-06-08 22:03:10 +02:00
Jonas Kvinge
b7c394b7a5 Disable fading when volume control is disabled 2019-06-08 12:04:04 +02:00
Jonas Kvinge
a070681f89 Move to private 2019-06-08 12:03:48 +02:00
Jonas Kvinge
d5e424eec8 Fix save/restore group by for internet collection 2019-06-08 00:15:54 +02:00
Jonas Kvinge
e0366d38f1 Change remove from favorites icon 2019-06-08 00:14:11 +02:00
Jonas Kvinge
cbf1d96b16 Avoid duplicate songs in the collection backend 2019-06-07 23:02:43 +02:00
Jonas Kvinge
7cc0d6bb5a Fix ItemFromSong() missing album id 2019-06-07 22:54:34 +02:00
Jonas Kvinge
e19b840ee6 Tidal artist id fixes 2019-06-07 21:02:28 +02:00
Jonas Kvinge
5c2ca1e3d9 Add tidal add/remove favorites + more tidal fixes 2019-06-07 20:23:05 +02:00
Jonas Kvinge
059c4beb30 Improve lyrics searcher 2019-06-06 18:22:41 +02:00
Jonas Kvinge
427808226f Change snap interface to network-manager-observe 2019-06-01 14:29:00 +02:00
Jonas Kvinge
5448245942 Update libprotobuf 2019-05-31 20:47:36 +02:00
Jonas Kvinge
34ab907007 Remove unused code 2019-05-31 20:47:15 +02:00
Jonas Kvinge
e5fde27859 Fix crash in tidal service 2019-05-31 02:32:01 +02:00
Jonas Kvinge
699e8e3ddb Fix schemas 2019-05-31 01:41:17 +02:00
Jonas Kvinge
28ee967371 Fix compile without xinescope 2019-05-31 01:29:56 +02:00
Jonas Kvinge
1b0b5f2554 Remove assert 2019-05-30 19:11:46 +02:00
Jonas Kvinge
1bcc86f989 Fix song search fetchalbums option 2019-05-30 18:46:17 +02:00
Jonas Kvinge
f26f932fd7 Queue tidal requests 2019-05-30 18:06:48 +02:00
Jonas Kvinge
2b7d48ce77 Distinguish albums with same name with album_id 2019-05-30 18:05:07 +02:00
Jonas Kvinge
111712fd6d Add artist_id, album_id and song id to songs 2019-05-30 18:04:30 +02:00
Jonas Kvinge
7609bc181e Implement offset to all Tidal requests 2019-05-29 17:39:51 +02:00
Jonas Kvinge
8b05af7ca3 Make xine analyzer optional 2019-05-28 18:37:51 +02:00
Jonas Kvinge
20f9108ebf Fix artists and albums limit 2019-05-28 17:34:44 +02:00
Jonas Kvinge
cecae7cdc3 Fix xine metronom compile 2019-05-27 21:56:07 +02:00
Jonas Kvinge
e5f3123567 Fix gst_pad_send_event block in ErrorMessageReceived() 2019-05-27 21:39:38 +02:00
Jonas Kvinge
27b0e27cfd initialize search id in the constructor. 2019-05-27 21:38:39 +02:00
Jonas Kvinge
890fba0f61 Add internet tabs view and tidal favorites (#167) 2019-05-27 21:10:37 +02:00
Jonas Kvinge
c4b732ff93 Add proper error handling to local redirectserver 2019-05-21 22:51:53 +02:00
Jonas Kvinge
b8fa2985d5 Fix Tidal SendLogin signal 2019-05-20 19:14:33 +02:00
Jonas Kvinge
ba741fbce8 Exclude version and path from singleapplication, dont start if message
could not be sent.
2019-05-20 18:41:12 +02:00
Jonas Kvinge
67d070c334 Increase default timeout 2019-05-20 18:40:34 +02:00
Jonas Kvinge
8576afe6f5 Fix return value of sendMessage() 2019-05-20 18:39:49 +02:00
Jonas Kvinge
36807fd376 Validate track duration 2019-05-14 19:07:02 +02:00
Jonas Kvinge
2ad1d27a59 Remove stream schema setting and add Add HI_RES option 2019-05-14 18:45:38 +02:00
Jonas Kvinge
7cc9c75d15 Make tidal token configurable 2019-05-13 23:49:09 +02:00
Jonas Kvinge
f33609bbf8 Compare artist and album case-insensitive 2019-05-13 22:38:24 +02:00
Jonas Kvinge
69eeb4b0f8 Move QSearchField to widgets and remove rest of qocoa 2019-05-08 23:34:44 +02:00
Jonas Kvinge
aa583ec1aa Set ContentTypeHeader for Tidal requests 2019-05-08 23:08:29 +02:00
Jonas Kvinge
2b9f153257 Add missing stage packages to snap 2019-05-07 21:04:52 +02:00
Jonas Kvinge
cb9b7fc917 Update appdata desktop file 2019-05-07 17:03:20 +02:00
Jonas Kvinge
74b36c8884 Remove some unneeded interfaces from the snap 2019-05-07 17:02:58 +02:00
Jonas Kvinge
89ff7d6dae Change api url of tidal cover provider 2019-05-06 18:18:31 +02:00
Jonas Kvinge
91df420cb4 Update Dockerfile 2019-05-05 23:51:59 +02:00
Jonas Kvinge
fc25c9f0bc Turn on git revision 2019-05-05 23:51:46 +02:00
Jonas Kvinge
12a27e3a8d Update snap version 2019-05-05 23:28:41 +02:00
1321 changed files with 123278 additions and 57564 deletions

603
.circleci/config.yml Normal file
View File

@@ -0,0 +1,603 @@
version: 2.1
commands:
cmake:
description: Configure build
steps:
- run:
name: Configure build
command: cmake ..
working_directory: build
build_source:
description: Create source tarball
steps:
- run:
name: Create source tarball
command: ../dist/scripts/maketarball.sh
working_directory: build
build_rpm:
description: Build RPM
steps:
- run:
name: Create RPM build sources directories
command: mkdir -p ~/rpmbuild/SOURCES /usr/src/packages/SOURCES
- run:
name: Copy source tarball 1
command: cp strawberry-*.tar.xz ~/rpmbuild/SOURCES/
working_directory: build
- run:
name: Copy source tarball 2
command: cp strawberry-*.tar.xz /usr/src/packages/SOURCES/
working_directory: build
- run:
name: Build RPM
command: rpmbuild -ba ../dist/unix/strawberry.spec
working_directory: build
build_deb:
description: Build Deb
steps:
- run:
name: make deb
command: dpkg-buildpackage -b -d -uc -us -nc -j2
install_opensuse_dependencies:
description: Install openSUSE dependencies
steps:
- run:
name: Update packages
command: zypper --non-interactive --gpg-auto-import-keys ref
- run:
name: Install openSUSE dependencies
command: >
zypper --non-interactive --gpg-auto-import-keys install
lsb-release
rpm-build
git
tar
make
cmake
gcc
gcc-c++
gettext-tools
glibc-devel
libboost_headers-devel
boost-devel
glib2-devel
glib2-tools
dbus-1-devel
alsa-devel
libnotify-devel
libgnutls-devel
protobuf-devel
sqlite3-devel
libpulse-devel
gstreamer-devel
gstreamer-plugins-base-devel
vlc-devel
libQt5Core-devel
libQt5Gui-devel
libQt5Widgets-devel
libQt5Concurrent-devel
libQt5Network-devel
libQt5Sql-devel
libQt5DBus-devel
libQt5Test-devel
libqt5-qtx11extras-devel
libqt5-qtbase-common-devel
libQt5Sql5-sqlite
libqt5-linguist-devel
libcdio-devel
libgpod-devel
libmtp-devel
libchromaprint-devel
desktop-file-utils
update-desktop-files
appstream-glib
hicolor-icon-theme
install_fedora_dependencies:
description: Install Fedora dependencies
steps:
- run:
name: Update packages
command: yum update --assumeyes
- run:
name: Upgrade packages
command: yum upgrade --assumeyes
- run:
name: Install Fedora dependencies
command: >
dnf install --assumeyes
@development-tools
redhat-lsb-core
git
glibc
gcc-c++
rpmdevtools
make
cmake
pkgconfig
glib
man
tar
gettext
openssh
boost-devel
dbus-devel
protobuf-devel
protobuf-compiler
sqlite-devel
alsa-lib-devel
pulseaudio-libs-devel
libnotify-devel
gnutls-devel
qt5-qtbase-devel
qt5-qtx11extras-devel
qt5-qttools-devel
gstreamer1-devel
gstreamer1-plugins-base-devel
taglib-devel
libcdio-devel
libgpod-devel
libmtp-devel
libchromaprint-devel
fftw-devel
desktop-file-utils
libappstream-glib
hicolor-icon-theme
install_centos_dependencies:
description: Install CentOS dependencies
steps:
- run:
name: Install epel-release
command: dnf install -y epel-release
- run:
name: Install epel-release-latest-8.noarch.rpm
command: dnf install -y https://dl.fedoraproject.org/pub/epel/epel-release-latest-8.noarch.rpm
- run:
name: Install config-manager
command: dnf install -y 'dnf-command(config-manager)'
- run:
name: PowerTools
command: dnf config-manager --set-enabled PowerTools
- run:
name: DNF Clean All
command: dnf clean all
- run:
name: Update packages
command: dnf update -y
- run:
name: Install CentOS dependencies
command: >
dnf install -y
glibc
gcc-c++
make
libtool
cmake3
rpmdevtools
redhat-lsb-core
git
man
tar
gettext
boost-devel
fuse-devel
dbus-devel
libnotify-devel
gnutls-devel
sqlite-devel
protobuf-devel
protobuf-compiler
alsa-lib-devel
pulseaudio-libs-devel
qt5-devel
qt5-qtbase-devel
qt5-qtx11extras-devel
qt5-qttools-devel
fftw-devel
libchromaprint-devel
libcdio-devel
libgpod-devel
libmtp-devel
libjpeg-devel
cairo-devel
dbus-x11
xorg-x11-server-Xvfb
xorg-x11-xauth
vim-common
desktop-file-utils
libappstream-glib
appstream-data
hicolor-icon-theme
python3-pip
python3-devel
gstreamer1-devel
gstreamer1-plugins-base-devel
install_mageia_dependencies:
description: Install Mageia dependencies
steps:
- run:
name: Update packages
command: urpmi.update --auto -a
- run:
name: Configure auto update
command: urpmi --auto --auto-update
- run:
name: Install dependencies
command: >
urpmi --auto --force
urpmi-debuginfo-install
git
tar
rpmdevtools
make
cmake
glibc
binutils
gcc-c++
man
gettext
notification-daemon
dbus-devel
libgnutls-devel
lib64boost-devel
lib64protobuf-devel
protobuf-compiler
lib64sqlite3-devel
lib64alsa2-devel
lib64pulseaudio-devel
lib64notify-devel
lib64qt5core-devel
lib64qt5gui-devel
lib64qt5widgets-devel
lib64qt5network-devel
lib64qt5concurrent-devel
lib64qt5sql-devel
lib64qt5dbus-devel
lib64qt5x11extras-devel
lib64qt5help-devel
libqt5test-devel
lib64gstreamer1.0-devel
lib64gstreamer-plugins-base1.0-devel
lib64cdio-devel
lib64gpod-devel
lib64mtp-devel
lib64raw1394-devel
lib64chromaprint-devel
libfftw-devel
desktop-file-utils
appstream-util
libappstream-glib8
hicolor-icon-theme
qt5ct
lib64mesaegl1
install_debian_dependencies:
description: Install Debian dependencies
steps:
- run:
name: Install Debian dependencies
command: >
apt-get update && apt-get install -y
build-essential
ssh
git
make
cmake
gcc
pkg-config
fakeroot
gettext
lsb-release
libglib2.0-dev
dpkg-dev
libdbus-1-dev
libboost-dev
libprotobuf-dev
protobuf-compiler
libsqlite3-dev
libgnutls28-dev
libasound2-dev
libpulse-dev
qtbase5-dev
qtbase5-dev-tools
qtbase5-private-dev
libqt5x11extras5-dev
qttools5-dev
libgstreamer1.0-dev
libgstreamer-plugins-base1.0-dev
gstreamer1.0-alsa
gstreamer1.0-pulseaudio
libchromaprint-dev
libfftw3-dev
libcdio-dev
libmtp-dev
libgpod-dev
install_ubuntu_dependencies:
description: Install Ubuntu dependencies
steps:
- run:
name: Setup environment
command: |
echo 'export DEBIAN_FRONTEND=noninteractive' >> $BASH_ENV
source $BASH_ENV
- run:
name: Install Ubuntu dependencies
command: >
apt-get update && apt-get install -y
build-essential
ssh
git
make
cmake
pkg-config
gcc
fakeroot
wget
curl
gettext
lsb-release
dpkg-dev
libglib2.0-dev
libboost-dev
libdbus-1-dev
libprotobuf-dev
protobuf-compiler
libsqlite3-dev
libgnutls28-dev
libasound2-dev
libpulse-dev
qtbase5-dev
qtbase5-dev-tools
qtbase5-private-dev
libqt5x11extras5-dev
qttools5-dev
libgstreamer1.0-dev
libgstreamer-plugins-base1.0-dev
libgstreamer-plugins-good1.0-dev
gstreamer1.0-alsa
gstreamer1.0-pulseaudio
libchromaprint-dev
libfftw3-dev
libcdio-dev
libmtp-dev
libgpod-dev
jobs:
build_source:
docker:
- image: opensuse/leap:15.1
steps:
- install_opensuse_dependencies
- checkout
- cmake
- build_source
build_opensuse_tumbleweed:
docker:
- image: opensuse/tumbleweed
environment:
RPM_BUILD_NCPUS: "2"
steps:
- run:
name: Update packages
command: zypper --non-interactive --gpg-auto-import-keys ref
- run:
name: Upgrade packages
command: zypper --non-interactive --gpg-auto-import-keys dup
- install_opensuse_dependencies
- checkout
- cmake
- build_source
- build_rpm
build_opensuse_lp151:
docker:
- image: opensuse/leap:15.1
environment:
RPM_BUILD_NCPUS: "2"
steps:
- install_opensuse_dependencies
- checkout
- cmake
- build_source
- build_rpm
build_opensuse_lp152:
docker:
- image: opensuse/leap:15.2
environment:
RPM_BUILD_NCPUS: "2"
steps:
- install_opensuse_dependencies
- checkout
- cmake
- build_source
- build_rpm
build_fedora_31:
docker:
- image: fedora:31
environment:
RPM_BUILD_NCPUS: "2"
steps:
- install_fedora_dependencies
- checkout
- cmake
- build_source
- build_rpm
build_fedora_32:
docker:
- image: fedora:32
environment:
RPM_BUILD_NCPUS: "2"
steps:
- install_fedora_dependencies
- checkout
- cmake
- build_source
- build_rpm
build_centos_8:
docker:
- image: centos:8
environment:
RPM_BUILD_NCPUS: "2"
steps:
- install_centos_dependencies
- checkout
- cmake
- build_source
- build_rpm
build_mageia_7:
docker:
- image: mageia:7
environment:
RPM_BUILD_NCPUS: "2"
steps:
- install_mageia_dependencies
- checkout
- cmake
- build_source
- build_rpm
build_debian_buster:
docker:
- image: debian:buster
steps:
- install_debian_dependencies
- checkout
- cmake
- build_deb
build_debian_bullseye:
docker:
- image: debian:bullseye
steps:
- install_debian_dependencies
- checkout
- cmake
- build_deb
build_ubuntu_bionic:
docker:
- image: ubuntu:bionic
steps:
- install_ubuntu_dependencies
- checkout
- cmake
- build_deb
build_ubuntu_focal:
docker:
- image: ubuntu:focal
steps:
- install_ubuntu_dependencies
- checkout
- cmake
- build_deb
build_ubuntu_groovy:
docker:
- image: ubuntu:groovy
steps:
- install_ubuntu_dependencies
- checkout
- cmake
- build_deb
workflows:
version: 2
build_all:
jobs:
- build_source:
filters:
tags:
only: /.*/
- build_opensuse_tumbleweed:
filters:
tags:
only: /.*/
- build_opensuse_lp151:
filters:
tags:
only: /.*/
- build_opensuse_lp152:
filters:
tags:
only: /.*/
- build_fedora_31:
filters:
tags:
only: /.*/
- build_fedora_32:
filters:
tags:
only: /.*/
- build_mageia_7:
filters:
tags:
only: /.*/
- build_centos_8:
filters:
tags:
only: /.*/
- build_debian_buster:
filters:
tags:
only: /.*/
- build_debian_bullseye:
filters:
tags:
only: /.*/
- build_ubuntu_bionic:
filters:
tags:
only: /.*/
- build_ubuntu_focal:
filters:
tags:
only: /.*/
- build_ubuntu_groovy:
filters:
tags:
only: /.*/

1
.github/FUNDING.yml vendored Normal file
View File

@@ -0,0 +1 @@
github: jonaski

29
.github/ISSUE_TEMPLATE/bug_report.md vendored Normal file
View File

@@ -0,0 +1,29 @@
---
name: Bug report
about: Create a report to help us improve
title: ''
labels: ''
assignees: ''
---
For general technical questions and help with technical issues please use the forum on https://forum.strawberrymusicplayer.org/
**Describe the bug**
A clear and concise description of what the bug is.
**To Reproduce**
Steps to reproduce the behavior.
**Expected behavior**
A clear and concise description of what you expected to happen.
**Screenshots:**
If applicable, add screenshots to help explain your problem.
**System Information:**
- Operating system:
- Strawberry Version:
**Additional context**
Add any other context about the problem here.

1130
.github/workflows/ccpp.yml vendored Normal file

File diff suppressed because it is too large Load Diff

6
.gitignore vendored
View File

@@ -73,6 +73,11 @@ CMakeLists.txt.user*
target_wrapper.* target_wrapper.*
compile_commands.json compile_commands.json
*.kdev4
*.vscode
*.code-workspace
*.sublime-workspace
# Temporary files # Temporary files
*~ *~
*.autosave *.autosave
@@ -99,7 +104,6 @@ Thumbs.db
# Stuff in dist # Stuff in dist
maketarball.sh maketarball.sh
create-dmg.sh
changelog changelog
PKGBUILD PKGBUILD

View File

@@ -1,54 +1,58 @@
sudo: required sudo: required
language: C++ language: C++
os: os:
- linux
- osx - osx
services: services:
- docker - docker
compiler: compiler:
- gcc - gcc
- clang
before_install: before_install:
- echo $DEPLOY_KEY_ENC | base64 --decode | openssl aes-256-cbc -K $encrypted_83a41ac424a6_key -iv $encrypted_83a41ac424a6_iv -out ~/.ssh/id_rsa -d - if ! [ "$DEPLOY_KEY_ENC" == "" ]; then
- chmod 600 ~/.ssh/id_rsa echo $DEPLOY_KEY_ENC | base64 --decode | openssl aes-256-cbc -K $encrypted_83a41ac424a6_key -iv $encrypted_83a41ac424a6_iv -out ~/.ssh/id_rsa -d ;
chmod 600 ~/.ssh/id_rsa ;
fi
- if [[ "$TRAVIS_OS_NAME" == "linux" ]]; then - if [[ "$TRAVIS_OS_NAME" == "linux" ]]; then
docker build -f Dockerfile -t strawberry-build .; docker build -f Dockerfile -t strawberry-build . || travis_terminate 1;
docker run --name build -itd strawberry-build /bin/bash; docker run --name build -itd strawberry-build /bin/bash || travis_terminate 1;
docker exec build git clone https://github.com/jonaski/strawberry; docker exec build git clone https://github.com/strawberrymusicplayer/strawberry;
fi fi
- if [[ "$TRAVIS_OS_NAME" == "osx" ]]; then - if [[ "$TRAVIS_OS_NAME" == "osx" ]]; then
git fetch --unshallow; git fetch --unshallow || travis_terminate 1;
git pull; git pull || travis_terminate 1;
brew update; brew unlink python@2 || travis_terminate 1;
brew unlink python; brew install glib pkgconfig libffi protobuf protobuf-c qt gettext gnutls fftw sqlite chromaprint zlib taglib;
brew install glib pkgconfig libffi protobuf protobuf-c qt gettext gnutls fftw; brew install gstreamer gst-plugins-base gst-plugins-good gst-plugins-bad gst-plugins-ugly gst-libav;
brew install sqlite --with-fts; brew install libcdio libmtp;
brew install gstreamer gst-plugins-base; brew install create-dmg;
brew install gst-plugins-good --with-flac; brew cask install sparkle;
brew install gst-plugins-bad gst-plugins-ugly gst-libav; sudo ln -s /usr/local/Caskroom/sparkle/$(ls /usr/local/Caskroom/sparkle | head -n1)/Sparkle.framework /Library/Frameworks/Sparkle.framework;
brew install chromaprint; sudo ln -s /usr/local/Caskroom/sparkle/$(ls /usr/local/Caskroom/sparkle | head -n1)/Sparkle.framework.dSYM /Library/Frameworks/Sparkle.framework.dSYM;
brew install libcdio libmtp libimobiledevice libplist;
export Qt5_DIR=/usr/local/opt/qt5/lib/cmake; export Qt5_DIR=/usr/local/opt/qt5/lib/cmake;
export Qt5LinguistTools_DIR=/usr/local/Cellar/qt/$(ls /usr/local/Cellar/qt | head -n1)/lib/cmake/Qt5LinguistTools; export Qt5LinguistTools_DIR=/usr/local/opt/qt5/lib/cmake/Qt5LinguistTools;
export PATH="/usr/local/opt/gettext/bin:$PATH"; export PATH="/usr/local/opt/gettext/bin:$PATH";
export PKG_CONFIG_PATH=/usr/local/Cellar/libffi/$(ls /usr/local/Cellar/libffi | head -n1)/lib/pkgconfig/:$PKG_CONFIG_PATH; export PKG_CONFIG_PATH="/usr/local/opt/libffi/lib/pkgconfig/:/usr/local/opt/zlib/lib/pkgconfig:$PKG_CONFIG_PATH";
ls /usr/local/lib/gstreamer-1.0; ls /usr/local/lib/gstreamer-1.0;
fi fi
before_script: before_script:
- if [[ "$TRAVIS_OS_NAME" == "linux" ]]; then docker exec build cmake -Hstrawberry -Bbuild -DENABLE_TRANSLATIONS=ON ; fi - if [[ "$TRAVIS_OS_NAME" == "linux" ]]; then docker exec build cmake -Hstrawberry -Bbuild ; fi
- if [[ "$TRAVIS_OS_NAME" == "osx" ]]; then mkdir build; cd build; cmake .. -DUSE_BUNDLE=ON -DENABLE_TRANSLATIONS=ON ; fi - if [[ "$TRAVIS_OS_NAME" == "osx" ]]; then mkdir build; cd build; cmake .. -DUSE_BUNDLE=ON ; fi
script: script:
- if [[ "$TRAVIS_OS_NAME" == "linux" ]]; then docker exec build make -C build -j8 ; fi - if [[ "$TRAVIS_OS_NAME" == "linux" ]]; then docker exec build make -C build -j8 ; fi
- if [[ "$TRAVIS_OS_NAME" == "osx" ]]; then - if [[ "$TRAVIS_OS_NAME" == "osx" ]]; then
make -j8; make -j8 || travis_terminate 1;
sudo make install; make install || travis_terminate 1;
sudo ../dist/macos/macdeploy.py strawberry.app; make dmg;
../dist/macos/create-dmg.sh strawberry.app $CC_FOR_BUILD;
fi fi
after_success: after_success:
- if [[ "$TRAVIS_OS_NAME" == "osx" ]]; then ls -lh strawberry.dmg; fi - if [[ "$TRAVIS_OS_NAME" == "osx" ]]; then ls -lh strawberry*.dmg; fi
- if [[ "$TRAVIS_OS_NAME" == "osx" ]] && [[ "$TRAVIS_BRANCH" == "master" ]]; then rsync -e "ssh -o StrictHostKeyChecking=no" -va strawberry*.dmg travis@echoes.jkvinge.net:/home/travis/builds/macos; fi - if [[ "$TRAVIS_OS_NAME" == "osx" ]] && [[ "$CC_FOR_BUILD" == "gcc" ]] && [ -f ~/.ssh/id_rsa ]; then
if [[ "$TRAVIS_BRANCH" == "master" ]]; then
rsync -e "ssh -o StrictHostKeyChecking=no" -va strawberry*.dmg travis@echoes.jkvinge.net:/home/travis/builds/macos;
elif [[ "$TRAVIS_BRANCH" == "macos" ]]; then
rsync -e "ssh -o StrictHostKeyChecking=no" -va strawberry*.dmg travis@echoes.jkvinge.net:/home/travis/builds/macos;
fi
fi
branches: branches:
except: except:

19
3rdparty/README.md vendored
View File

@@ -19,7 +19,7 @@ TagLib is a library for reading and editing the meta-data of several popular aud
by Strawberry to identify audio files. It is important that it is kept up-to-date for Strawberry to function by Strawberry to identify audio files. It is important that it is kept up-to-date for Strawberry to function
correctly. correctly.
It is kept in 3rdparty because there currently is no offical release of TagLib with the features and bugfixes It is kept in 3rdparty because there currently is no official release of TagLib with the features and bugfixes
that are in the official repository. And also because some distros use older, or unpatched versions. that are in the official repository. And also because some distros use older, or unpatched versions.
There is a bug in the latest version (1.11.1) corrupting Ogg files, There is a bug in the latest version (1.11.1) corrupting Ogg files,
@@ -37,21 +37,6 @@ URL: https://github.com/taglib/taglib
utf8-cpp utf8-cpp
-------- --------
This is 2 header files used by taglib, but kept in a seperate directory because it is maintained by others. This is 2 header files used by taglib, but kept in a separate directory because it is maintained by others.
URL: http://utfcpp.sourceforge.net/ URL: http://utfcpp.sourceforge.net/
qocoa
-----
This is a small static library currently used for the search fields above the collection, playlist and in
the cover manager. It is slightly modified from original version.
URL: https://github.com/mikemcquaid/Qocoa
SPMediaKeyTap
-------------
This is used for macOS only to enable strawberry to grab global shortcuts and can safely be deleted on other
platforms.

View File

@@ -1,13 +0,0 @@
set(SPMEDIAKEY-SOURCES
SPMediaKeyTap.m
SPInvocationGrabbing/NSObject+SPInvocationGrabbing.m
)
set(SPMEDIAKEY-HEADERS
SPMediaKeyTap.h
SPInvocationGrabbing/NSObject+SPInvocationGrabbing.h
)
ADD_LIBRARY(SPMediaKeyTap STATIC
${SPMEDIAKEY-SOURCES}
)

View File

@@ -1,8 +0,0 @@
Copyright (c) 2011, Joachim Bengtsson
All rights reserved.
Redistribution and use in source and binary forms, with or without modification, are permitted provided that the following conditions are met:
* Neither the name of the organization nor the names of its contributors may be used to endorse or promote products derived from this software without specific prior written permission.
THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT HOLDER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.

View File

@@ -1,12 +0,0 @@
SPMediaKeyTap
=============
`SPMediaKeyTap` abstracts a `CGEventHook` and other nastiness in order to give you a relatively simple API to receive media key events (prev/next/playpause, on F7 to F9 on modern MacBook Pros) exclusively, without them reaching other applications like iTunes. `SPMediaKeyTap` is clever enough to resign its exclusive lock on media keys by looking for which application was active most recently: if that application is in `SPMediaKeyTap`'s whitelist, it will resign the keys. This is similar to the behavior of Apple's applications collaborating on media key handling exclusivity, but unfortunately, Apple are not exposing any APIs allowing third-parties to join in on this collaboration.
For now, the whitelist is just a hardcoded array in `+[SPMediaKeyTap defaultMediaKeyUserBundleIdentifiers]`. If your app starts using `SPMediaKeyTap`, please [mail me](mailto:nevyn@spotify.com) your bundle ID, and I'll include it in the canonical repository. This is a bad solution; a better solution would be to use distributed notifications to collaborate in creating this whitelist at runtime. Hopefully someone'll have the time and energy to write this soon.
In `Example/SPMediaKeyTapExampleAppDelegate.m` is an example of both how you use `SPMediaKeyTap`, and how you handle the semi-private `NSEvent` subtypes involved in media keys, including on how to fall back to non-event tap handling of these events.
`SPMediaKeyTap` and other `CGEventHook`s on the event type `NSSystemDefined` is known to interfere with each other and applications doing weird stuff with mouse input, because mouse clicks are also part of the `NSSystemDefined` category. The single issue we have had reported here at Spotify is Adobe Fireworks, in which item selection stops working with `SPMediaKeyTap` is active.
`SPMediaKeyTap` requires 10.5 to work, and disables itself on 10.4.

View File

@@ -1,30 +0,0 @@
#import <Foundation/Foundation.h>
@interface SPInvocationGrabber : NSObject {
id _object;
NSInvocation *_invocation;
int frameCount;
char **frameStrings;
BOOL backgroundAfterForward;
BOOL onMainAfterForward;
BOOL waitUntilDone;
}
-(id)initWithObject:(id)obj;
-(id)initWithObject:(id)obj stacktraceSaving:(BOOL)saveStack;
@property (readonly, retain, nonatomic) id object;
@property (readonly, retain, nonatomic) NSInvocation *invocation;
@property BOOL backgroundAfterForward;
@property BOOL onMainAfterForward;
@property BOOL waitUntilDone;
-(void)invoke; // will release object and invocation
-(void)printBacktrace;
-(void)saveBacktrace;
@end
@interface NSObject (SPInvocationGrabbing)
-(id)grab;
-(id)invokeAfter:(NSTimeInterval)delta;
-(id)nextRunloop;
-(id)inBackground;
-(id)onMainAsync:(BOOL)async;
@end

View File

@@ -1,128 +0,0 @@
#import "NSObject+SPInvocationGrabbing.h"
#import <execinfo.h>
#pragma mark Invocation grabbing
@interface SPInvocationGrabber ()
@property (readwrite, retain, nonatomic) id object;
@property (readwrite, retain, nonatomic) NSInvocation *invocation;
@end
@implementation SPInvocationGrabber
- (id)initWithObject:(id)obj;
{
return [self initWithObject:obj stacktraceSaving:YES];
}
-(id)initWithObject:(id)obj stacktraceSaving:(BOOL)saveStack;
{
self.object = obj;
if(saveStack)
[self saveBacktrace];
return self;
}
-(void)dealloc;
{
free(frameStrings);
self.object = nil;
self.invocation = nil;
[super dealloc];
}
@synthesize invocation = _invocation, object = _object;
@synthesize backgroundAfterForward, onMainAfterForward, waitUntilDone;
- (void)runInBackground;
{
NSAutoreleasePool *pool = [[NSAutoreleasePool alloc] init];
@try {
[self invoke];
}
@finally {
[pool drain];
}
}
- (void)forwardInvocation:(NSInvocation *)anInvocation {
[anInvocation retainArguments];
anInvocation.target = _object;
self.invocation = anInvocation;
if(backgroundAfterForward)
[NSThread detachNewThreadSelector:@selector(runInBackground) toTarget:self withObject:nil];
else if(onMainAfterForward)
[self performSelectorOnMainThread:@selector(invoke) withObject:nil waitUntilDone:waitUntilDone];
}
- (NSMethodSignature *)methodSignatureForSelector:(SEL)inSelector {
NSMethodSignature *signature = [super methodSignatureForSelector:inSelector];
if (signature == NULL)
signature = [_object methodSignatureForSelector:inSelector];
return signature;
}
- (void)invoke;
{
@try {
[_invocation invoke];
}
@catch (NSException * e) {
NSLog(@"SPInvocationGrabber's target raised %@:\n\t%@\nInvocation was originally scheduled at:", e.name, e);
[self printBacktrace];
printf("\n");
[e raise];
}
self.invocation = nil;
self.object = nil;
}
-(void)saveBacktrace;
{
void *backtraceFrames[128];
frameCount = backtrace(&backtraceFrames[0], 128);
frameStrings = backtrace_symbols(&backtraceFrames[0], frameCount);
}
-(void)printBacktrace;
{
int x;
for(x = 3; x < frameCount; x++) {
if(frameStrings[x] == NULL) { break; }
printf("%s\n", frameStrings[x]);
}
}
@end
@implementation NSObject (SPInvocationGrabbing)
-(id)grab;
{
return [[[SPInvocationGrabber alloc] initWithObject:self] autorelease];
}
-(id)invokeAfter:(NSTimeInterval)delta;
{
id grabber = [self grab];
[NSTimer scheduledTimerWithTimeInterval:delta target:grabber selector:@selector(invoke) userInfo:nil repeats:NO];
return grabber;
}
- (id)nextRunloop;
{
return [self invokeAfter:0];
}
-(id)inBackground;
{
SPInvocationGrabber *grabber = [self grab];
grabber.backgroundAfterForward = YES;
return grabber;
}
-(id)onMainAsync:(BOOL)async;
{
SPInvocationGrabber *grabber = [self grab];
grabber.onMainAfterForward = YES;
grabber.waitUntilDone = !async;
return grabber;
}
@end

View File

@@ -1,28 +0,0 @@
// A
+(UIView*)flashAt:(CGRect)r in:(UIView*)parent color:(UIColor*)color;
{
float duration = 0.5;
UIView *flash = [[[UIView alloc] initWithFrame:r] autorelease];
flash.backgroundColor = color;
[parent addSubview:flash];
[[flash invokeAfter:duration+0.1] removeFromSuperview];
[UIView beginAnimations:@"SPFlash" context:NULL];
[UIView setAnimationDuration:duration];
flash.alpha = 0.0;
[UIView commitAnimations];
return flash;
}
// B
- (void)tableView:(UITableView *)tableView didSelectRowAtIndexPath:(NSIndexPath *)indexPath {
// Force the animation to happen by calling this method again after a small
// delay - see http://blog.instapaper.com/post/53568356
[[self nextRunloop] delayedTableViewDidSelectRowAtIndexPath: indexPath];
}
// C
[[tableView invokeAfter:0.15] selectRowAtIndexPath:indexPath animated:YES scrollPosition:UITableViewScrollPositionNone];
[[tableView invokeAfter:0.30] deselectRowAtIndexPath:indexPath animated:YES];
[[tableView invokeAfter:0.45] selectRowAtIndexPath:indexPath animated:YES scrollPosition:UITableViewScrollPositionNone];

View File

@@ -1,12 +0,0 @@
@interface MyClass : NSObject
-(BOOL)areTheNewViewersGoneYet:(Duck*)duck;
@end
...
MyClass *myInstance = [[MyClass alloc] init];
id invocationGrabber = [[[SPInvocationGrabber alloc] initWithTarget:myInstance] autorelease];
[invocationGrabber areTheNewViewersGoneYet:[Duck yellowDuck]]; // line 9
NSInvocation *invocationForAreTheNewViewersGoneYet = [invocationGrabber invocation];

View File

@@ -1,38 +0,0 @@
#import <Cocoa/Cocoa.h>
#import "NSObject+SPInvocationGrabbing.h"
@interface Foo : NSObject {
int a;
}
-(void)startIt;
-(void)theBackgroundStuff;
-(void)theForegroundStuff;
@end
@implementation Foo
-(void)startIt;
{
NSLog(@"Starting out on the main thread...");
a = 3;
[[self inBackground] theBackgroundStuff];
}
-(void)theBackgroundStuff;
{
NSLog(@"Woah, this is a background thread!");
a += 6;
[[self onMainAsync:YES] theForegroundStuff];
}
-(void)theForegroundStuff;
{
NSLog(@"Hey presto: %d", a);
}
@end
int main() {
NSAutoreleasePool *pool = [NSAutoreleasePool new];
Foo *foo = [Foo new];
[foo startIt];
[[NSRunLoop currentRunLoop] runUntilDate:[NSDate dateWithTimeIntervalSinceNow:1]];
[pool release];
return 0;
}

View File

@@ -1,34 +0,0 @@
#include <Cocoa/Cocoa.h>
#import <IOKit/hidsystem/ev_keymap.h>
#import <Carbon/Carbon.h>
// http://overooped.com/post/2593597587/mediakeys
#define SPSystemDefinedEventMediaKeys 8
@interface SPMediaKeyTap : NSObject {
EventHandlerRef _app_switching_ref;
EventHandlerRef _app_terminating_ref;
CFMachPortRef _eventPort;
CFRunLoopSourceRef _eventPortSource;
CFRunLoopRef _tapThreadRL;
BOOL _shouldInterceptMediaKeyEvents;
id _delegate;
// The app that is frontmost in this list owns media keys
NSMutableArray *_mediaKeyAppList;
}
+ (NSArray*)defaultMediaKeyUserBundleIdentifiers;
-(id)initWithDelegate:(id)delegate;
+(BOOL)usesGlobalMediaKeyTap;
-(void)startWatchingMediaKeys;
-(void)stopWatchingMediaKeys;
-(void)handleAndReleaseMediaKeyEvent:(NSEvent *)event;
@end
@interface NSObject (SPMediaKeyTapDelegate)
-(void)mediaKeyTap:(SPMediaKeyTap*)keyTap receivedMediaKeyEvent:(NSEvent*)event;
@end
extern NSString *kMediaKeyUsingBundleIdentifiersDefaultsKey;

View File

@@ -1,300 +0,0 @@
// Copyright (c) 2010 Spotify AB
#import "SPMediaKeyTap.h"
#import "SPInvocationGrabbing/NSObject+SPInvocationGrabbing.h" // https://gist.github.com/511181, in submodule
@interface SPMediaKeyTap ()
-(BOOL)shouldInterceptMediaKeyEvents;
-(void)setShouldInterceptMediaKeyEvents:(BOOL)newSetting;
-(void)startWatchingAppSwitching;
-(void)stopWatchingAppSwitching;
-(void)eventTapThread;
@end
static SPMediaKeyTap *singleton = nil;
static pascal OSStatus appSwitched (EventHandlerCallRef nextHandler, EventRef evt, void* userData);
static pascal OSStatus appTerminated (EventHandlerCallRef nextHandler, EventRef evt, void* userData);
static CGEventRef tapEventCallback(CGEventTapProxy proxy, CGEventType type, CGEventRef event, void *refcon);
// Inspired by http://gist.github.com/546311
@implementation SPMediaKeyTap
#pragma mark -
#pragma mark Setup and teardown
-(id)initWithDelegate:(id)delegate;
{
_delegate = delegate;
[self startWatchingAppSwitching];
singleton = self;
_mediaKeyAppList = [NSMutableArray new];
return self;
}
-(void)dealloc;
{
[self stopWatchingMediaKeys];
[self stopWatchingAppSwitching];
[_mediaKeyAppList release];
[super dealloc];
}
-(void)startWatchingAppSwitching;
{
// Listen to "app switched" event, so that we don't intercept media keys if we
// weren't the last "media key listening" app to be active
EventTypeSpec eventType = { kEventClassApplication, kEventAppFrontSwitched };
OSStatus err = InstallApplicationEventHandler(NewEventHandlerUPP(appSwitched), 1, &eventType, self, &_app_switching_ref);
assert(err == noErr);
eventType.eventKind = kEventAppTerminated;
err = InstallApplicationEventHandler(NewEventHandlerUPP(appTerminated), 1, &eventType, self, &_app_terminating_ref);
assert(err == noErr);
}
-(void)stopWatchingAppSwitching;
{
if(!_app_switching_ref) return;
RemoveEventHandler(_app_switching_ref);
_app_switching_ref = NULL;
}
-(void)startWatchingMediaKeys;{
[self setShouldInterceptMediaKeyEvents:YES];
// Add an event tap to intercept the system defined media key events
_eventPort = CGEventTapCreate(kCGSessionEventTap,
kCGHeadInsertEventTap,
kCGEventTapOptionDefault,
CGEventMaskBit(NX_SYSDEFINED),
tapEventCallback,
self);
assert(_eventPort != NULL);
_eventPortSource = CFMachPortCreateRunLoopSource(kCFAllocatorSystemDefault, _eventPort, 0);
assert(_eventPortSource != NULL);
// Let's do this in a separate thread so that a slow app doesn't lag the event tap
[NSThread detachNewThreadSelector:@selector(eventTapThread) toTarget:self withObject:nil];
}
-(void)stopWatchingMediaKeys;
{
// TODO<nevyn>: Shut down thread, remove event tap port and source
}
#pragma mark -
#pragma mark Accessors
+(BOOL)usesGlobalMediaKeyTap
{
#ifdef _DEBUG
// breaking in gdb with a key tap inserted sometimes locks up all mouse and keyboard input forever, forcing reboot
return NO;
#else
// XXX(nevyn): MediaKey event tap doesn't work on 10.4, feel free to figure out why if you have the energy.
return floor(NSAppKitVersionNumber) >= 949/*NSAppKitVersionNumber10_5*/;
#endif
}
+ (NSArray*)defaultMediaKeyUserBundleIdentifiers;
{
return [NSArray arrayWithObjects:
[[NSBundle mainBundle] bundleIdentifier], // your app
@"com.spotify.client",
@"com.apple.iTunes",
@"com.apple.QuickTimePlayerX",
@"com.apple.quicktimeplayer",
@"com.apple.iWork.Keynote",
@"com.apple.iPhoto",
@"org.videolan.vlc",
@"com.apple.Aperture",
@"com.plexsquared.Plex",
@"com.soundcloud.desktop",
@"com.macromedia.fireworks", // the tap messes up their mouse input
nil
];
}
-(BOOL)shouldInterceptMediaKeyEvents;
{
BOOL shouldIntercept = NO;
@synchronized(self) {
shouldIntercept = _shouldInterceptMediaKeyEvents;
}
return shouldIntercept;
}
-(void)pauseTapOnTapThread:(BOOL)yeahno;
{
CGEventTapEnable(self->_eventPort, yeahno);
}
-(void)setShouldInterceptMediaKeyEvents:(BOOL)newSetting;
{
BOOL oldSetting;
@synchronized(self) {
oldSetting = _shouldInterceptMediaKeyEvents;
_shouldInterceptMediaKeyEvents = newSetting;
}
if(_tapThreadRL && oldSetting != newSetting) {
id grab = [self grab];
[grab pauseTapOnTapThread:newSetting];
NSTimer *timer = [NSTimer timerWithTimeInterval:0 invocation:[grab invocation] repeats:NO];
CFRunLoopAddTimer(_tapThreadRL, (CFRunLoopTimerRef)timer, kCFRunLoopCommonModes);
}
}
#pragma mark
#pragma mark -
#pragma mark Event tap callbacks
// Note: method called on background thread
static CGEventRef tapEventCallback2(CGEventTapProxy proxy, CGEventType type, CGEventRef event, void *refcon)
{
SPMediaKeyTap *self = refcon;
if(type == kCGEventTapDisabledByTimeout) {
NSLog(@"Media key event tap was disabled by timeout");
CGEventTapEnable(self->_eventPort, TRUE);
return event;
} else if(type == kCGEventTapDisabledByUserInput) {
// Was disabled manually by -[pauseTapOnTapThread]
return event;
}
NSEvent *nsEvent = nil;
@try {
nsEvent = [NSEvent eventWithCGEvent:event];
}
@catch (NSException * e) {
NSLog(@"Strange CGEventType: %d: %@", type, e);
assert(0);
return event;
}
if (type != NX_SYSDEFINED || [nsEvent subtype] != SPSystemDefinedEventMediaKeys)
return event;
int keyCode = (([nsEvent data1] & 0xFFFF0000) >> 16);
if (keyCode != NX_KEYTYPE_PLAY && keyCode != NX_KEYTYPE_FAST && keyCode != NX_KEYTYPE_REWIND)
return event;
if (![self shouldInterceptMediaKeyEvents])
return event;
[nsEvent retain]; // matched in handleAndReleaseMediaKeyEvent:
[self performSelectorOnMainThread:@selector(handleAndReleaseMediaKeyEvent:) withObject:nsEvent waitUntilDone:NO];
return NULL;
}
static CGEventRef tapEventCallback(CGEventTapProxy proxy, CGEventType type, CGEventRef event, void *refcon)
{
NSAutoreleasePool *pool = [NSAutoreleasePool new];
CGEventRef ret = tapEventCallback2(proxy, type, event, refcon);
[pool drain];
return ret;
}
// event will have been retained in the other thread
-(void)handleAndReleaseMediaKeyEvent:(NSEvent *)event {
[event autorelease];
[_delegate mediaKeyTap:self receivedMediaKeyEvent:event];
}
-(void)eventTapThread;
{
_tapThreadRL = CFRunLoopGetCurrent();
CFRunLoopAddSource(_tapThreadRL, _eventPortSource, kCFRunLoopCommonModes);
CFRunLoopRun();
}
#pragma mark Task switching callbacks
NSString *kMediaKeyUsingBundleIdentifiersDefaultsKey = @"SPApplicationsNeedingMediaKeys";
-(void)mediaKeyAppListChanged;
{
if([_mediaKeyAppList count] == 0) return;
/*NSLog(@"--");
int i = 0;
for (NSValue *psnv in _mediaKeyAppList) {
ProcessSerialNumber psn; [psnv getValue:&psn];
NSDictionary *processInfo = [(id)ProcessInformationCopyDictionary(
&psn,
kProcessDictionaryIncludeAllInformationMask
) autorelease];
NSString *bundleIdentifier = [processInfo objectForKey:(id)kCFBundleIdentifierKey];
NSLog(@"%d: %@", i++, bundleIdentifier);
}*/
ProcessSerialNumber mySerial, topSerial;
GetCurrentProcess(&mySerial);
[[_mediaKeyAppList objectAtIndex:0] getValue:&topSerial];
Boolean same;
OSErr err = SameProcess(&mySerial, &topSerial, &same);
[self setShouldInterceptMediaKeyEvents:(err == noErr && same)];
}
-(void)appIsNowFrontmost:(ProcessSerialNumber)psn;
{
NSValue *psnv = [NSValue valueWithBytes:&psn objCType:@encode(ProcessSerialNumber)];
NSDictionary *processInfo = [(id)ProcessInformationCopyDictionary(
&psn,
kProcessDictionaryIncludeAllInformationMask
) autorelease];
NSString *bundleIdentifier = [processInfo objectForKey:(id)kCFBundleIdentifierKey];
NSArray *whitelistIdentifiers = [[NSUserDefaults standardUserDefaults] arrayForKey:kMediaKeyUsingBundleIdentifiersDefaultsKey];
if(![whitelistIdentifiers containsObject:bundleIdentifier]) return;
[_mediaKeyAppList removeObject:psnv];
[_mediaKeyAppList insertObject:psnv atIndex:0];
[self mediaKeyAppListChanged];
}
-(void)appTerminated:(ProcessSerialNumber)psn;
{
NSValue *psnv = [NSValue valueWithBytes:&psn objCType:@encode(ProcessSerialNumber)];
[_mediaKeyAppList removeObject:psnv];
[self mediaKeyAppListChanged];
}
static pascal OSStatus appSwitched (EventHandlerCallRef nextHandler, EventRef evt, void* userData)
{
SPMediaKeyTap *self = (id)userData;
ProcessSerialNumber newSerial;
GetFrontProcess(&newSerial);
[self appIsNowFrontmost:newSerial];
return CallNextEventHandler(nextHandler, evt);
}
static pascal OSStatus appTerminated (EventHandlerCallRef nextHandler, EventRef evt, void* userData)
{
SPMediaKeyTap *self = (id)userData;
ProcessSerialNumber deadPSN;
GetEventParameter(
evt,
kEventParamProcessID,
typeProcessSerialNumber,
NULL,
sizeof(deadPSN),
NULL,
&deadPSN
);
[self appTerminated:deadPSN];
return CallNextEventHandler(nextHandler, evt);
}
@end

View File

@@ -1,25 +0,0 @@
-(void)mediaKeyTap:(SPMediaKeyTap*)keyTap receivedMediaKeyEvent:(NSEvent*)event;
{
assert([event type] == NSSystemDefined && [event subtype] == SPSystemDefinedEventMediaKeys);
int keyCode = (([event data1] & 0xFFFF0000) >> 16);
int keyFlags = ([event data1] & 0x0000FFFF);
int keyState = (((keyFlags & 0xFF00) >> 8)) == 0xA;
int keyRepeat = (keyFlags & 0x1);
if (keyState == 1 && windowController != NULL) {
switch (keyCode) {
case NX_KEYTYPE_PLAY:
... return;
case NX_KEYTYPE_FAST:
... return;
case NX_KEYTYPE_REWIND:
... return;
}
}
}

View File

@@ -1,38 +0,0 @@
cmake_minimum_required(VERSION 2.8.11)
set(CMAKE_C_FLAGS "${CMAKE_C_FLAGS} -Wall")
set(CMAKE_CXX_STANDARD 11)
set(CMAKE_CXX_FLAGS "${CMAKE_CXX_FLAGS} --std=c++11 -U__STRICT_ANSI__")
set(SOURCES)
set(HEADERS
qsearchfield.h
qbutton.h
qprogressindicatorspinning.h
)
qt5_wrap_cpp(MOC_SOURCES ${HEADERS})
if(APPLE)
list(APPEND SOURCES
qsearchfield_mac.mm
qbutton_mac.mm
qprogressindicatorspinning_mac.mm
)
else()
list(APPEND SOURCES
qsearchfield_nonmac.cpp
qbutton_nonmac.cpp
qprogressindicatorspinning_nonmac.cpp
)
set(RESOURCES
qsearchfield_nonmac.qrc
qprogressindicatorspinning_nonmac.qrc
)
qt5_add_resources(RESOURCES_SOURCES ${RESOURCES})
endif()
add_library(Qocoa STATIC ${SOURCES} ${MOC_SOURCES} ${RESOURCES_SOURCES})
target_link_libraries(Qocoa ${QT_LIBRARIES})

View File

@@ -1,19 +0,0 @@
Copyright (C) 2011 by Mike McQuaid
Permission is hereby granted, free of charge, to any person obtaining a copy
of this software and associated documentation files (the "Software"), to deal
in the Software without restriction, including without limitation the rights
to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
copies of the Software, and to permit persons to whom the Software is
furnished to do so, subject to the following conditions:
The above copyright notice and this permission notice shall be included in
all copies or substantial portions of the Software.
THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN
THE SOFTWARE.

View File

@@ -1,38 +0,0 @@
# Qocoa
Qocoa is a collection of Qt wrappers for OSX's Cocoa widgets.
## Features
- basic fallback to sensible Qt types on non-OSX platforms
- shared class headers which expose no implementation details
- typical Qt signal/slot-based API
- trivial to import into projects (class header/implementation, [single shared global header](https://github.com/mikemcquaid/Qocoa/blob/master/qocoa_mac.h))
## Building
```
git clone git://github.com/mikemcquaid/Qocoa.git
cd Qocoa
qmake # or cmake .
make
```
## Status
I'm not personally working on this any more but will accept pull-requests.
[![Build Status](https://travis-ci.org/MikeMcQuaid/Qocoa.svg?branch=master)](https://travis-ci.org/MikeMcQuaid/Qocoa)
## Usage
For each class you want to use copy the [`qocoa_mac.h`](https://github.com/mikemcquaid/Qocoa/blob/master/qocoa_mac.h), `$CLASS.h`, `$CLASS_mac.*` and `$CLASS_nonmac.*` files into your source tree and add them to your buildsystem. Examples are provided for [CMake](https://github.com/mikemcquaid/Qocoa/blob/master/CMakeLists.txt) and [QMake](https://github.com/mikemcquaid/Qocoa/blob/master/Qocoa.pro).
## Contact
[Mike McQuaid](mailto:mike@mikemcquaid.com)
## License
Qocoa is licensed under the [MIT License](http://en.wikipedia.org/wiki/MIT_License).
The full license text is available in [LICENSE.txt](https://github.com/mikemcquaid/Qocoa/blob/master/LICENSE.txt).
Magnifier and EditClear icons taken from [QtCreator](http://qt-project.org/) and are licensed under the [LGPL](http://www.gnu.org/copyleft/lesser.html).
Other icons are taken from the [Oxygen Project](http://www.oxygen-icons.org/) and are licensed under the [Creative Commons Attribution-ShareAlike 3.0 License](http://creativecommons.org/licenses/by-sa/3.0/).
## Gallery
![Qocoa Gallery](https://github.com/mikemcquaid/Qocoa/raw/master/gallery.png)

View File

@@ -1,13 +0,0 @@
Widgets I hope to implement (or at least investigate):
- NSTokenField
- NSSegmentedControl
- NSLevelIndicator
- NSPathControl
- NSSlider (Circular)
- NSSplitView
- NSTextFinder
- NSOutlineView in an NSScrollView (Source List)
- NSDrawer
- PDFView
- WebView

View File

@@ -1,49 +0,0 @@
#ifndef QBUTTON_H
#define QBUTTON_H
#include <QWidget>
#include <QPointer>
class QButtonPrivate;
class QButton : public QWidget
{
Q_OBJECT
public:
// Matches NSBezelStyle
enum BezelStyle {
Rounded = 1,
RegularSquare = 2,
Disclosure = 5,
ShadowlessSquare = 6,
Circular = 7,
TexturedSquare = 8,
HelpButton = 9,
SmallSquare = 10,
TexturedRounded = 11,
RoundRect = 12,
Recessed = 13,
RoundedDisclosure = 14,
#ifdef __MAC_10_7
Inline = 15
#endif
};
explicit QButton(QWidget *parent, BezelStyle bezelStyle = Rounded);
public slots:
void setText(const QString &text);
void setImage(const QPixmap &image);
void setChecked(bool checked);
public:
void setCheckable(bool checkable);
bool isChecked();
signals:
void clicked(bool checked = false);
private:
friend class QButtonPrivate;
QPointer<QButtonPrivate> pimpl;
};
#endif // QBUTTON_H

View File

@@ -1,229 +0,0 @@
/*
Copyright (C) 2011 by Mike McQuaid
Permission is hereby granted, free of charge, to any person obtaining a copy
of this software and associated documentation files (the "Software"), to deal
in the Software without restriction, including without limitation the rights
to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
copies of the Software, and to permit persons to whom the Software is
furnished to do so, subject to the following conditions:
The above copyright notice and this permission notice shall be included in
all copies or substantial portions of the Software.
THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN
THE SOFTWARE.
*/
#include "qbutton.h"
#include "qocoa_mac.h"
#import "Foundation/NSAutoreleasePool.h"
#import "AppKit/NSButton.h"
#import "AppKit/NSFont.h"
class QButtonPrivate : public QObject
{
public:
QButtonPrivate(QButton *qButton, NSButton *nsButton, QButton::BezelStyle bezelStyle)
: QObject(qButton), qButton(qButton), nsButton(nsButton)
{
switch(bezelStyle) {
case QButton::Disclosure:
case QButton::Circular:
#ifdef __MAC_10_7
case QButton::Inline:
#endif
case QButton::RoundedDisclosure:
case QButton::HelpButton:
[nsButton setTitle:@""];
default:
break;
}
NSFont* font = 0;
switch(bezelStyle) {
case QButton::RoundRect:
font = [NSFont fontWithName:@"Lucida Grande" size:12];
break;
case QButton::Recessed:
font = [NSFont fontWithName:@"Lucida Grande Bold" size:12];
break;
#ifdef __MAC_10_7
case QButton::Inline:
font = [NSFont boldSystemFontOfSize:[NSFont systemFontSizeForControlSize:NSSmallControlSize]];
break;
#endif
default:
font = [NSFont systemFontOfSize:[NSFont systemFontSizeForControlSize:NSRegularControlSize]];
break;
}
[nsButton setFont:font];
switch(bezelStyle) {
case QButton::Rounded:
qButton->setMinimumWidth(40);
qButton->setFixedHeight(24);
qButton->setSizePolicy(QSizePolicy::Minimum, QSizePolicy::Fixed);
break;
case QButton::RegularSquare:
case QButton::TexturedSquare:
qButton->setMinimumSize(14, 23);
qButton->setSizePolicy(QSizePolicy::Minimum, QSizePolicy::Minimum);
break;
case QButton::ShadowlessSquare:
qButton->setMinimumSize(5, 25);
qButton->setSizePolicy(QSizePolicy::Minimum, QSizePolicy::Minimum);
break;
case QButton::SmallSquare:
qButton->setMinimumSize(4, 21);
qButton->setSizePolicy(QSizePolicy::Minimum, QSizePolicy::Minimum);
break;
case QButton::TexturedRounded:
qButton->setMinimumSize(10, 22);
qButton->setSizePolicy(QSizePolicy::Minimum, QSizePolicy::Minimum);
break;
case QButton::RoundRect:
case QButton::Recessed:
qButton->setMinimumWidth(16);
qButton->setFixedHeight(18);
qButton->setSizePolicy(QSizePolicy::Minimum, QSizePolicy::Fixed);
break;
case QButton::Disclosure:
qButton->setMinimumWidth(13);
qButton->setFixedHeight(13);
qButton->setSizePolicy(QSizePolicy::Minimum, QSizePolicy::Fixed);
break;
case QButton::Circular:
qButton->setMinimumSize(16, 16);
qButton->setMaximumHeight(40);
qButton->setSizePolicy(QSizePolicy::Minimum, QSizePolicy::Minimum);
break;
case QButton::HelpButton:
case QButton::RoundedDisclosure:
qButton->setMinimumWidth(22);
qButton->setFixedHeight(22);
qButton->setSizePolicy(QSizePolicy::Minimum, QSizePolicy::Fixed);
break;
#ifdef __MAC_10_7
case QButton::Inline:
qButton->setMinimumWidth(10);
qButton->setFixedHeight(16);
qButton->setSizePolicy(QSizePolicy::Minimum, QSizePolicy::Fixed);
break;
#endif
}
switch(bezelStyle) {
case QButton::Recessed:
[nsButton setButtonType:NSPushOnPushOffButton];
case QButton::Disclosure:
[nsButton setButtonType:NSOnOffButton];
default:
[nsButton setButtonType:NSMomentaryPushInButton];
}
[nsButton setBezelStyle:(__bridge NSBezelStyle)bezelStyle];
}
void clicked()
{
emit qButton->clicked(qButton->isChecked());
}
~QButtonPrivate() {
[[nsButton target] release];
[nsButton setTarget:nil];
}
QButton *qButton;
NSButton *nsButton;
};
@interface QButtonTarget : NSObject
{
@public
QPointer<QButtonPrivate> pimpl;
}
-(void)clicked;
@end
@implementation QButtonTarget
-(void)clicked {
Q_ASSERT(pimpl);
if (pimpl)
pimpl->clicked();
}
@end
QButton::QButton(QWidget *parent, BezelStyle bezelStyle) : QWidget(parent)
{
NSAutoreleasePool *pool = [[NSAutoreleasePool alloc] init];
NSButton *button = [[NSButton alloc] init];
pimpl = new QButtonPrivate(this, button, bezelStyle);
QButtonTarget *target = [[QButtonTarget alloc] init];
target->pimpl = pimpl;
[button setTarget:target];
[button setAction:@selector(clicked)];
setupLayout(button, this);
[button release];
[pool drain];
}
void QButton::setText(const QString &text)
{
Q_ASSERT(pimpl);
if (!pimpl)
return;
NSAutoreleasePool *pool = [[NSAutoreleasePool alloc] init];
[pimpl->nsButton setTitle:fromQString(text)];
[pool drain];
}
void QButton::setImage(const QPixmap &image)
{
Q_ASSERT(pimpl);
if (pimpl)
[pimpl->nsButton setImage:fromQPixmap(image)];
}
void QButton::setChecked(bool checked)
{
Q_ASSERT(pimpl);
if (pimpl)
[pimpl->nsButton setState:checked];
}
void QButton::setCheckable(bool checkable)
{
const NSInteger cellMask = checkable ? NSChangeBackgroundCellMask : NSNoCellMask;
Q_ASSERT(pimpl);
if (pimpl)
[[pimpl->nsButton cell] setShowsStateBy:cellMask];
}
bool QButton::isChecked()
{
Q_ASSERT(pimpl);
if (!pimpl)
return false;
return [pimpl->nsButton state];
}

View File

@@ -1,89 +0,0 @@
/*
Copyright (C) 2011 by Mike McQuaid
Permission is hereby granted, free of charge, to any person obtaining a copy
of this software and associated documentation files (the "Software"), to deal
in the Software without restriction, including without limitation the rights
to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
copies of the Software, and to permit persons to whom the Software is
furnished to do so, subject to the following conditions:
The above copyright notice and this permission notice shall be included in
all copies or substantial portions of the Software.
THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN
THE SOFTWARE.
*/
#include "qbutton.h"
#include <QToolBar>
#include <QToolButton>
#include <QPushButton>
#include <QVBoxLayout>
class QButtonPrivate : public QObject
{
public:
QButtonPrivate(QButton *button, QAbstractButton *abstractButton)
: QObject(button), abstractButton(abstractButton) {}
QPointer<QAbstractButton> abstractButton;
};
QButton::QButton(QWidget *parent, BezelStyle) : QWidget(parent)
{
QAbstractButton *button = 0;
if (qobject_cast<QToolBar*>(parent))
button = new QToolButton(this);
else
button = new QPushButton(this);
connect(button, SIGNAL(clicked()),
this, SIGNAL(clicked()));
pimpl = new QButtonPrivate(this, button);
QVBoxLayout *layout = new QVBoxLayout(this);
layout->setMargin(0);
layout->addWidget(button);
}
void QButton::setText(const QString &text)
{
Q_ASSERT(pimpl);
if (pimpl)
pimpl->abstractButton->setText(text);
}
void QButton::setImage(const QPixmap &image)
{
Q_ASSERT(pimpl);
if (pimpl)
pimpl->abstractButton->setIcon(image);
}
void QButton::setChecked(bool checked)
{
Q_ASSERT(pimpl);
if (pimpl)
pimpl->abstractButton->setChecked(checked);
}
void QButton::setCheckable(bool checkable)
{
Q_ASSERT(pimpl);
if (pimpl)
pimpl->abstractButton->setCheckable(checkable);
}
bool QButton::isChecked()
{
Q_ASSERT(pimpl);
if (!pimpl)
return false;
return pimpl->abstractButton->isChecked();
}

View File

@@ -1,29 +0,0 @@
#ifndef QPROGRESSINDICATORSPINNING_H
#define QPROGRESSINDICATORSPINNING_H
#include <QWidget>
#include <QPointer>
class QProgressIndicatorSpinningPrivate;
class QProgressIndicatorSpinning : public QWidget
{
Q_OBJECT
public:
// Matches NSProgressIndicatorThickness
enum Thickness {
Default = 14,
Small = 10,
Large = 18,
Aqua = 12
};
explicit QProgressIndicatorSpinning(QWidget *parent,
Thickness thickness = Default);
public slots:
void animate(bool animate = true);
private:
friend class QProgressIndicatorSpinningPrivate;
QPointer<QProgressIndicatorSpinningPrivate> pimpl;
};
#endif // QPROGRESSINDICATORSPINNING_H

View File

@@ -1,70 +0,0 @@
/*
Copyright (C) 2011 by Mike McQuaid
Permission is hereby granted, free of charge, to any person obtaining a copy
of this software and associated documentation files (the "Software"), to deal
in the Software without restriction, including without limitation the rights
to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
copies of the Software, and to permit persons to whom the Software is
furnished to do so, subject to the following conditions:
The above copyright notice and this permission notice shall be included in
all copies or substantial portions of the Software.
THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN
THE SOFTWARE.
*/
#include "qprogressindicatorspinning.h"
#include "qocoa_mac.h"
#import "Foundation/NSAutoreleasePool.h"
#import "AppKit/NSProgressIndicator.h"
class QProgressIndicatorSpinningPrivate : public QObject
{
public:
QProgressIndicatorSpinningPrivate(QProgressIndicatorSpinning *qProgressIndicatorSpinning,
NSProgressIndicator *nsProgressIndicator)
: QObject(qProgressIndicatorSpinning), nsProgressIndicator(nsProgressIndicator) {}
NSProgressIndicator *nsProgressIndicator;
};
QProgressIndicatorSpinning::QProgressIndicatorSpinning(QWidget *parent,
Thickness thickness)
: QWidget(parent)
{
NSAutoreleasePool *pool = [[NSAutoreleasePool alloc] init];
NSProgressIndicator *progress = [[NSProgressIndicator alloc] init];
[progress setStyle:NSProgressIndicatorSpinningStyle];
pimpl = new QProgressIndicatorSpinningPrivate(this, progress);
setupLayout(progress, this);
setFixedSize(thickness, thickness);
[progress release];
[pool drain];
}
void QProgressIndicatorSpinning::animate(bool animate)
{
Q_ASSERT(pimpl);
if (!pimpl)
return;
if (animate)
[pimpl->nsProgressIndicator startAnimation:nil];
else
[pimpl->nsProgressIndicator stopAnimation:nil];
}

View File

@@ -1,72 +0,0 @@
/*
Copyright (C) 2011 by Mike McQuaid
Permission is hereby granted, free of charge, to any person obtaining a copy
of this software and associated documentation files (the "Software"), to deal
in the Software without restriction, including without limitation the rights
to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
copies of the Software, and to permit persons to whom the Software is
furnished to do so, subject to the following conditions:
The above copyright notice and this permission notice shall be included in
all copies or substantial portions of the Software.
THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN
THE SOFTWARE.
*/
#include "qprogressindicatorspinning.h"
#include <QVBoxLayout>
#include <QMovie>
#include <QLabel>
class QProgressIndicatorSpinningPrivate : public QObject
{
public:
QProgressIndicatorSpinningPrivate(QProgressIndicatorSpinning *qProgressIndicatorSpinning,
QMovie *movie)
: QObject(qProgressIndicatorSpinning), movie(movie) {}
QPointer<QMovie> movie;
};
QProgressIndicatorSpinning::QProgressIndicatorSpinning(QWidget *parent,
Thickness thickness)
: QWidget(parent)
{
QVBoxLayout *layout = new QVBoxLayout(this);
layout->setMargin(0);
QSize size(thickness, thickness);
QMovie *movie = new QMovie(this);
movie->setFileName(":/Qocoa/qprogressindicatorspinning_nonmac.gif");
movie->setScaledSize(size);
// Roughly match OSX speed.
movie->setSpeed(200);
pimpl = new QProgressIndicatorSpinningPrivate(this, movie);
QLabel *label = new QLabel(this);
label->setMovie(movie);
layout->addWidget(label);
setFixedSize(size);
}
void QProgressIndicatorSpinning::animate(bool animate)
{
Q_ASSERT(pimpl && pimpl->movie);
if (!(pimpl && pimpl->movie))
return;
if (animate)
pimpl->movie->start();
else
pimpl->movie->stop();
}

Binary file not shown.

Before

Width:  |  Height:  |  Size: 3.1 KiB

View File

@@ -1,5 +0,0 @@
<RCC>
<qresource prefix="/Qocoa">
<file>qprogressindicatorspinning_nonmac.gif</file>
</qresource>
</RCC>

View File

@@ -1,7 +0,0 @@
<RCC>
<qresource prefix="/Qocoa">
<file>qsearchfield_nonmac_clear.png</file>
<file>qsearchfield_nonmac_magnifier_menu.png</file>
<file>qsearchfield_nonmac_magnifier.png</file>
</qresource>
</RCC>

Binary file not shown.

Before

Width:  |  Height:  |  Size: 736 B

Binary file not shown.

Before

Width:  |  Height:  |  Size: 300 B

Binary file not shown.

Before

Width:  |  Height:  |  Size: 439 B

View File

@@ -1,17 +1,55 @@
cmake_minimum_required(VERSION 2.8.11) cmake_minimum_required(VERSION 3.0)
set(CMAKE_C_FLAGS "${CMAKE_C_FLAGS} -Wall") include(CheckIncludeFiles)
set(CMAKE_CXX_STANDARD 11) include(CheckFunctionExists)
set(CMAKE_CXX_FLAGS "${CMAKE_CXX_FLAGS} --std=c++11 -U__STRICT_ANSI__ -Wall -Woverloaded-virtual -Wno-sign-compare -Wno-deprecated-declarations -Wno-unused-local-typedefs -fpermissive")
if(CMAKE_VERSION VERSION_GREATER 3.0)
check_function_exists(geteuid HAVE_GETEUID)
check_function_exists(getpwuid HAVE_GETPWUID)
endif()
set(SINGLEAPP-SOURCES singleapplication.cpp singleapplication_p.cpp) set(SINGLEAPP-SOURCES singleapplication.cpp singleapplication_p.cpp)
set(SINGLEAPP-MOC-HEADERS singleapplication.h singleapplication_p.h) set(SINGLEAPP-MOC-HEADERS singleapplication.h singleapplication_p.h)
QT5_WRAP_CPP(SINGLEAPP-SOURCES-MOC ${SINGLEAPP-MOC-HEADERS}) if(WITH_QT6)
ADD_LIBRARY(singleapplication STATIC ${SINGLEAPP-SOURCES} ${SINGLEAPP-SOURCES-MOC}) qt6_wrap_cpp(SINGLEAPP-SOURCES-MOC ${SINGLEAPP-MOC-HEADERS})
target_link_libraries(singleapplication Qt5::Core Qt5::Widgets Qt5::Network) else()
qt5_wrap_cpp(SINGLEAPP-SOURCES-MOC ${SINGLEAPP-MOC-HEADERS})
endif()
add_library(singleapplication STATIC ${SINGLEAPP-SOURCES} ${SINGLEAPP-SOURCES-MOC})
target_include_directories(singleapplication SYSTEM PRIVATE
${QtCore_INCLUDE_DIRS}
${QtWidgets_INCLUDE_DIRS}
${QtNetwork_INCLUDE_DIRS}
)
target_include_directories(singleapplication PRIVATE
${CMAKE_CURRENT_SOURCE_DIR}
${CMAKE_CURRENT_BINARY_DIR}
)
target_link_libraries(singleapplication PRIVATE
${QtCore_LIBRARIES}
${QtWidgets_LIBRARIES}
${QtNetwork_LIBRARIES}
)
set(SINGLECOREAPP-SOURCES singlecoreapplication.cpp singlecoreapplication_p.cpp) set(SINGLECOREAPP-SOURCES singlecoreapplication.cpp singlecoreapplication_p.cpp)
set(SINGLECOREAPP-MOC-HEADERS singlecoreapplication.h singlecoreapplication_p.h) set(SINGLECOREAPP-MOC-HEADERS singlecoreapplication.h singlecoreapplication_p.h)
QT5_WRAP_CPP(SINGLECOREAPP-SOURCES-MOC ${SINGLECOREAPP-MOC-HEADERS}) if(WITH_QT6)
ADD_LIBRARY(singlecoreapplication STATIC ${SINGLECOREAPP-SOURCES} ${SINGLECOREAPP-SOURCES-MOC}) qt6_wrap_cpp(SINGLECOREAPP-SOURCES-MOC ${SINGLECOREAPP-MOC-HEADERS})
target_link_libraries(singlecoreapplication Qt5::Core Qt5::Widgets Qt5::Network) else()
qt5_wrap_cpp(SINGLECOREAPP-SOURCES-MOC ${SINGLECOREAPP-MOC-HEADERS})
endif()
add_library(singlecoreapplication STATIC ${SINGLECOREAPP-SOURCES} ${SINGLECOREAPP-SOURCES-MOC})
target_include_directories(singlecoreapplication SYSTEM PRIVATE
${QtCore_INCLUDE_DIRS}
${QtNetwork_INCLUDE_DIRS}
)
target_include_directories(singlecoreapplication PRIVATE
${CMAKE_CURRENT_SOURCE_DIR}
${CMAKE_CURRENT_BINARY_DIR}
)
target_link_libraries(singlecoreapplication PRIVATE
${QtCore_LIBRARIES}
${QtNetwork_LIBRARIES}
)
configure_file(config.h.in "${CMAKE_CURRENT_BINARY_DIR}/config.h")

View File

@@ -0,0 +1,2 @@
#cmakedefine HAVE_GETEUID
#cmakedefine HAVE_GETPWUID

View File

@@ -20,156 +20,178 @@
// OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN // OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN
// THE SOFTWARE. // THE SOFTWARE.
#include <QApplication> //
#include <QtCore/QTime> // W A R N I N G !!!
#include <QtCore/QThread> // -----------------
#include <QtCore/QDateTime> //
#include <QtCore/QByteArray> // This is a modified version of SingleApplication,
#include <QtCore/QSharedMemory> // The original version is at:
//
// https://github.com/itay-grudev/SingleApplication
//
//
#include <cstdlib>
#include <limits>
#include <QtGlobal>
#include <QCoreApplication>
#include <QThread>
#include <QSharedMemory>
#include <QLocalSocket>
#include <QByteArray>
#include <QElapsedTimer>
#include <QtDebug>
#if QT_VERSION >= QT_VERSION_CHECK(5, 10, 0)
# include <QRandomGenerator>
#else
# include <QDateTime>
#endif
#include "singleapplication.h" #include "singleapplication.h"
#include "singleapplication_p.h" #include "singleapplication_p.h"
/** /**
* @brief Constructor. Checks and fires up LocalServer or closes the program * @brief Constructor.
* if another instance already exists * Checks and fires up LocalServer or closes the program if another instance already exists
* @param argc * @param argc
* @param argv * @param argv
* @param {bool} allowSecondaryInstances * @param {bool} allowSecondaryInstances
*/ */
SingleApplication::SingleApplication( int &argc, char *argv[], bool allowSecondary, Options options, int timeout ) SingleApplication::SingleApplication(int &argc, char *argv[], bool allowSecondary, Options options, int timeout)
: app_t( argc, argv ), d_ptr( new SingleApplicationPrivate( this ) ) : app_t(argc, argv), d_ptr(new SingleApplicationPrivate(this)) {
{
Q_D(SingleApplication);
// Store the current mode of the program Q_D(SingleApplication);
d->options = options;
// Generating an application ID used for identifying the shared memory // Store the current mode of the program
// block and QLocalServer d->options = options;
d->genBlockServerName();
// Generating an application ID used for identifying the shared memory block and QLocalServer
d->genBlockServerName();
#ifdef Q_OS_UNIX #ifdef Q_OS_UNIX
// By explicitly attaching it and then deleting it we make sure that the // By explicitly attaching it and then deleting it we make sure that the
// memory is deleted even after the process has crashed on Unix. // memory is deleted even after the process has crashed on Unix.
d->memory = new QSharedMemory( d->blockServerName ); d->memory = new QSharedMemory(d->blockServerName);
d->memory->attach(); d->memory->attach();
delete d->memory; delete d->memory;
#endif #endif
// Guarantee thread safe behaviour with a shared memory block. // Guarantee thread safe behaviour with a shared memory block.
d->memory = new QSharedMemory( d->blockServerName ); d->memory = new QSharedMemory(d->blockServerName);
// Create a shared memory block // Create a shared memory block
if( d->memory->create( sizeof( InstancesInfo ) ) ) { if (d->memory->create(sizeof(InstancesInfo))) {
// Initialize the shared memory block // Initialize the shared memory block
d->memory->lock(); d->memory->lock();
d->initializeMemoryBlock(); d->initializeMemoryBlock();
d->memory->unlock(); d->memory->unlock();
} else { }
// Attempt to attach to the memory segment else {
if( ! d->memory->attach() ) { // Attempt to attach to the memory segment
qCritical() << "SingleApplication: Unable to attach to shared memory block."; if (! d->memory->attach()) {
qCritical() << d->memory->errorString(); qCritical() << "SingleApplication: Unable to attach to shared memory block.";
delete d; qCritical() << d->memory->errorString();
::exit( EXIT_FAILURE ); delete d;
} ::exit(EXIT_FAILURE);
} }
}
InstancesInfo* inst = static_cast<InstancesInfo*>( d->memory->data() ); InstancesInfo* inst = static_cast<InstancesInfo*>(d->memory->data());
QTime time; QElapsedTimer time;
time.start(); time.start();
// Make sure the shared memory block is initialised and in consistent state // Make sure the shared memory block is initialised and in consistent state
while( true ) { while (true) {
d->memory->lock(); d->memory->lock();
if( d->blockChecksum() == inst->checksum ) break; if (d->blockChecksum() == inst->checksum) break;
if( time.elapsed() > 5000 ) { if (time.elapsed() > 5000) {
qWarning() << "SingleApplication: Shared memory block has been in an inconsistent state from more than 5s. Assuming primary instance failure."; qWarning() << "SingleApplication: Shared memory block has been in an inconsistent state from more than 5s. Assuming primary instance failure.";
d->initializeMemoryBlock(); d->initializeMemoryBlock();
}
d->memory->unlock();
// Random sleep here limits the probability of a collision between two racing apps
qsrand( QDateTime::currentMSecsSinceEpoch() % std::numeric_limits<uint>::max() );
QThread::sleep( 8 + static_cast <unsigned long>( static_cast <float>( qrand() ) / RAND_MAX * 10 ) );
}
if( inst->primary == false) {
d->startPrimary();
d->memory->unlock();
return;
}
// Check if another instance can be started
if( allowSecondary ) {
inst->secondary += 1;
inst->checksum = d->blockChecksum();
d->instanceNumber = inst->secondary;
d->startSecondary();
if( d->options & Mode::SecondaryNotification ) {
d->connectToPrimary( timeout, SingleApplicationPrivate::SecondaryInstance );
}
d->memory->unlock();
return;
} }
d->memory->unlock(); d->memory->unlock();
d->connectToPrimary( timeout, SingleApplicationPrivate::NewInstance ); // Random sleep here limits the probability of a collision between two racing apps
#if QT_VERSION >= QT_VERSION_CHECK(5, 10, 0)
QThread::sleep(QRandomGenerator::global()->bounded(8u, 18u));
#else
qsrand(QDateTime::currentMSecsSinceEpoch() % std::numeric_limits<uint>::max());
QThread::sleep(8 + static_cast<unsigned long>(static_cast <float>(qrand()) / RAND_MAX * 10));
#endif
}
delete d; if (inst->primary == false) {
d->startPrimary();
d->memory->unlock();
return;
}
// Check if another instance can be started
if (allowSecondary) {
inst->secondary += 1;
inst->checksum = d->blockChecksum();
d->instanceNumber = inst->secondary;
d->startSecondary();
if (d->options & Mode::SecondaryNotification) {
d->connectToPrimary(timeout, SingleApplicationPrivate::SecondaryInstance);
}
d->memory->unlock();
return;
}
d->memory->unlock();
d->connectToPrimary(timeout, SingleApplicationPrivate::NewInstance);
delete d;
::exit(EXIT_SUCCESS);
::exit( EXIT_SUCCESS );
} }
/** /**
* @brief Destructor * @brief Destructor
*/ */
SingleApplication::~SingleApplication() SingleApplication::~SingleApplication() {
{ Q_D(SingleApplication);
Q_D(SingleApplication); delete d;
delete d;
} }
bool SingleApplication::isPrimary() bool SingleApplication::isPrimary() {
{ Q_D(SingleApplication);
Q_D(SingleApplication); return d->server != nullptr;
return d->server != nullptr;
} }
bool SingleApplication::isSecondary() bool SingleApplication::isSecondary() {
{ Q_D(SingleApplication);
Q_D(SingleApplication); return d->server == nullptr;
return d->server == nullptr;
} }
quint32 SingleApplication::instanceId() quint32 SingleApplication::instanceId() {
{ Q_D(SingleApplication);
Q_D(SingleApplication); return d->instanceNumber;
return d->instanceNumber;
} }
qint64 SingleApplication::primaryPid() qint64 SingleApplication::primaryPid() {
{ Q_D(SingleApplication);
Q_D(SingleApplication); return d->primaryPid();
return d->primaryPid();
} }
bool SingleApplication::sendMessage( QByteArray message, int timeout ) bool SingleApplication::sendMessage(QByteArray message, int timeout) {
{
Q_D(SingleApplication);
// Nobody to connect to Q_D(SingleApplication);
if( isPrimary() ) return false;
// Make sure the socket is connected // Nobody to connect to
d->connectToPrimary( timeout, SingleApplicationPrivate::Reconnect ); if (isPrimary()) return false;
// Make sure the socket is connected
d->connectToPrimary(timeout, SingleApplicationPrivate::Reconnect);
d->socket->write(message);
bool dataWritten = d->socket->waitForBytesWritten(timeout);
d->socket->flush();
return dataWritten;
d->socket->write( message );
bool dataWritten = d->socket->flush();
d->socket->waitForBytesWritten( timeout );
return dataWritten;
} }

View File

@@ -20,12 +20,24 @@
// OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN // OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN
// THE SOFTWARE. // THE SOFTWARE.
//
// W A R N I N G !!!
// -----------------
//
// This is a modified version of SingleApplication,
// The original version is at:
//
// https://github.com/itay-grudev/SingleApplication
//
//
#ifndef SINGLEAPPLICATION_H #ifndef SINGLEAPPLICATION_H
#define SINGLEAPPLICATION_H #define SINGLEAPPLICATION_H
#include <QtCore/QtGlobal> #include <QtGlobal>
#include <QApplication> #include <QApplication>
#include <QtNetwork/QLocalSocket> #include <QFlags>
#include <QByteArray>
class SingleApplicationPrivate; class SingleApplicationPrivate;
@@ -34,95 +46,95 @@ class SingleApplicationPrivate;
* Application * Application
* @see QCoreApplication * @see QCoreApplication
*/ */
class SingleApplication : public QApplication class SingleApplication : public QApplication {
{ Q_OBJECT
Q_OBJECT
typedef QApplication app_t; typedef QApplication app_t;
public: public:
/** /**
* @brief Mode of operation of SingleApplication. * @brief Mode of operation of SingleApplication.
* Whether the block should be user-wide or system-wide and whether the * Whether the block should be user-wide or system-wide and whether the
* primary instance should be notified when a secondary instance had been * primary instance should be notified when a secondary instance had been
* started. * started.
* @note Operating system can restrict the shared memory blocks to the same * @note Operating system can restrict the shared memory blocks to the same
* user, in which case the User/System modes will have no effect and the * user, in which case the User/System modes will have no effect and the
* block will be user wide. * block will be user wide.
* @enum * @enum
*/ */
enum Mode { enum Mode {
User = 1 << 0, User = 1 << 0,
System = 1 << 1, System = 1 << 1,
SecondaryNotification = 1 << 2, SecondaryNotification = 1 << 2,
ExcludeAppVersion = 1 << 3, ExcludeAppVersion = 1 << 3,
ExcludeAppPath = 1 << 4 ExcludeAppPath = 1 << 4
}; };
Q_DECLARE_FLAGS(Options, Mode) Q_DECLARE_FLAGS(Options, Mode)
/** /**
* @brief Intitializes a SingleApplication instance with argc command line * @brief Intitializes a SingleApplication instance with argc command line
* arguments in argv * arguments in argv
* @arg {int &} argc - Number of arguments in argv * @arg {int &} argc - Number of arguments in argv
* @arg {const char *[]} argv - Supplied command line arguments * @arg {const char *[]} argv - Supplied command line arguments
* @arg {bool} allowSecondary - Whether to start the instance as secondary * @arg {bool} allowSecondary - Whether to start the instance as secondary
* if there is already a primary instance. * if there is already a primary instance.
* @arg {Mode} mode - Whether for the SingleApplication block to be applied * @arg {Mode} mode - Whether for the SingleApplication block to be applied
* User wide or System wide. * User wide or System wide.
* @arg {int} timeout - Timeout to wait in miliseconds. * @arg {int} timeout - Timeout to wait in milliseconds.
* @note argc and argv may be changed as Qt removes arguments that it * @note argc and argv may be changed as Qt removes arguments that it
* recognizes * recognizes
* @note Mode::SecondaryNotification only works if set on both the primary * @note Mode::SecondaryNotification only works if set on both the primary
* instance and the secondary instance. * instance and the secondary instance.
* @note The timeout is just a hint for the maximum time of blocking * @note The timeout is just a hint for the maximum time of blocking
* operations. It does not guarantee that the SingleApplication * operations. It does not guarantee that the SingleApplication
* initialisation will be completed in given time, though is a good hint. * initialisation will be completed in given time, though is a good hint.
* Usually 4*timeout would be the worst case (fail) scenario. * Usually 4*timeout would be the worst case (fail) scenario.
* @see See the corresponding QAPPLICATION_CLASS constructor for reference * @see See the corresponding QAPPLICATION_CLASS constructor for reference
*/ */
explicit SingleApplication( int &argc, char *argv[], bool allowSecondary = false, Options options = Mode::User, int timeout = 1000 ); explicit SingleApplication( int &argc, char *argv[], bool allowSecondary = false, Options options = Mode::User, int timeout = 1000 );
~SingleApplication(); ~SingleApplication() override;
/** /**
* @brief Returns if the instance is the primary instance * @brief Returns if the instance is the primary instance
* @returns {bool} * @returns {bool}
*/ */
bool isPrimary(); bool isPrimary();
/** /**
* @brief Returns if the instance is a secondary instance * @brief Returns if the instance is a secondary instance
* @returns {bool} * @returns {bool}
*/ */
bool isSecondary(); bool isSecondary();
/** /**
* @brief Returns a unique identifier for the current instance * @brief Returns a unique identifier for the current instance
* @returns {qint32} * @returns {qint32}
*/ */
quint32 instanceId(); quint32 instanceId();
/** /**
* @brief Returns the process ID (PID) of the primary instance * @brief Returns the process ID (PID) of the primary instance
* @returns {qint64} * @returns {qint64}
*/ */
qint64 primaryPid(); qint64 primaryPid();
/** /**
* @brief Sends a message to the primary instance. Returns true on success. * @brief Sends a message to the primary instance. Returns true on success.
* @param {int} timeout - Timeout for connecting * @param {int} timeout - Timeout for connecting
* @returns {bool} * @returns {bool}
* @note sendMessage() will return false if invoked from the primary * @note sendMessage() will return false if invoked from the primary
* instance. * instance.
*/ */
bool sendMessage( QByteArray message, int timeout = 100 ); bool sendMessage( QByteArray message, int timeout = 1000 );
Q_SIGNALS: signals:
void instanceStarted(); void instanceStarted();
void receivedMessage( quint32 instanceId, QByteArray message ); void receivedMessage( quint32 instanceId, QByteArray message );
private:
SingleApplicationPrivate *d_ptr;
Q_DECLARE_PRIVATE(SingleApplication)
private:
SingleApplicationPrivate *d_ptr;
Q_DECLARE_PRIVATE(SingleApplication)
}; };
Q_DECLARE_OPERATORS_FOR_FLAGS(SingleApplication::Options) Q_DECLARE_OPERATORS_FOR_FLAGS(SingleApplication::Options)

View File

@@ -24,381 +24,378 @@
// W A R N I N G !!! // W A R N I N G !!!
// ----------------- // -----------------
// //
// This file is not part of the SingleApplication API. It is used purely as an // This is a modified version of SingleApplication,
// implementation detail. This header file may change from version to // The original version is at:
// version without notice, or may even be removed.
// //
// https://github.com/itay-grudev/SingleApplication
//
//
#include "config.h"
#include <QtGlobal>
#include <cstdlib> #include <cstdlib>
#include <cstddef> #include <cstddef>
#include <QtCore/QDir> #ifdef Q_OS_UNIX
#include <QtCore/QProcess> # include <unistd.h>
#include <QtCore/QByteArray> # include <sys/types.h>
#include <QtCore/QSemaphore> # include <pwd.h>
#include <QtCore/QDataStream> #endif
#include <QtCore/QStandardPaths>
#include <QtCore/QCryptographicHash> #include <QIODevice>
#include <QtNetwork/QLocalServer> #include <QSharedMemory>
#include <QtNetwork/QLocalSocket> #include <QByteArray>
#include <QDataStream>
#include <QCryptographicHash>
#include <QLocalServer>
#include <QLocalSocket>
#include <QDir>
#include "singleapplication.h" #include "singleapplication.h"
#include "singleapplication_p.h" #include "singleapplication_p.h"
#ifdef Q_OS_WIN #ifdef Q_OS_WIN
#include <windows.h> # include <windows.h>
#include <lmcons.h> # include <lmcons.h>
#endif #endif
SingleApplicationPrivate::SingleApplicationPrivate( SingleApplication *q_ptr ) SingleApplicationPrivate::SingleApplicationPrivate(SingleApplication *_q_ptr)
: q_ptr( q_ptr ) : q_ptr(_q_ptr),
{ memory(nullptr),
server = nullptr; socket(nullptr),
socket = nullptr; server(nullptr),
memory = nullptr; instanceNumber(-1)
instanceNumber = -1; {}
}
SingleApplicationPrivate::~SingleApplicationPrivate() SingleApplicationPrivate::~SingleApplicationPrivate() {
{
if( socket != nullptr ) {
socket->close();
delete socket;
}
memory->lock(); if (socket != nullptr) {
InstancesInfo* inst = static_cast<InstancesInfo*>(memory->data()); socket->close();
if( server != nullptr ) { delete socket;
server->close(); }
delete server;
inst->primary = false;
inst->primaryPid = -1;
inst->checksum = blockChecksum();
}
memory->unlock();
delete memory; memory->lock();
} InstancesInfo* inst = static_cast<InstancesInfo*>(memory->data());
if (server != nullptr) {
void SingleApplicationPrivate::genBlockServerName() server->close();
{ delete server;
QCryptographicHash appData( QCryptographicHash::Sha256 );
appData.addData( "SingleApplication", 17 );
appData.addData( SingleApplication::app_t::applicationName().toUtf8() );
appData.addData( SingleApplication::app_t::organizationName().toUtf8() );
appData.addData( SingleApplication::app_t::organizationDomain().toUtf8() );
if( ! (options & SingleApplication::Mode::ExcludeAppVersion) ) {
appData.addData( SingleApplication::app_t::applicationVersion().toUtf8() );
}
if( ! (options & SingleApplication::Mode::ExcludeAppPath) ) {
#ifdef Q_OS_WIN
appData.addData( SingleApplication::app_t::applicationFilePath().toLower().toUtf8() );
#else
appData.addData( SingleApplication::app_t::applicationFilePath().toUtf8() );
#endif
}
// User level block requires a user specific data in the hash
if( options & SingleApplication::Mode::User ) {
#ifdef Q_OS_WIN
wchar_t username [ UNLEN + 1 ];
// Specifies size of the buffer on input
DWORD usernameLength = UNLEN + 1;
if( GetUserNameW( username, &usernameLength ) ) {
appData.addData( QString::fromWCharArray(username).toUtf8() );
} else {
appData.addData( QStandardPaths::standardLocations( QStandardPaths::HomeLocation ).join("").toUtf8() );
}
#endif
#ifdef Q_OS_UNIX
QProcess process;
process.start( "whoami" );
if( process.waitForFinished( 100 ) &&
process.exitCode() == QProcess::NormalExit) {
appData.addData( process.readLine() );
} else {
appData.addData(
QDir(
QStandardPaths::standardLocations( QStandardPaths::HomeLocation ).first()
).absolutePath().toUtf8()
);
}
#endif
}
// Replace the backslash in RFC 2045 Base64 [a-zA-Z0-9+/=] to comply with
// server naming requirements.
blockServerName = appData.result().toBase64().replace("/", "_");
}
void SingleApplicationPrivate::initializeMemoryBlock()
{
InstancesInfo* inst = static_cast<InstancesInfo*>( memory->data() );
inst->primary = false; inst->primary = false;
inst->secondary = 0;
inst->primaryPid = -1; inst->primaryPid = -1;
inst->checksum = blockChecksum(); inst->checksum = blockChecksum();
}
memory->unlock();
delete memory;
} }
void SingleApplicationPrivate::startPrimary() void SingleApplicationPrivate::genBlockServerName() {
{
Q_Q(SingleApplication);
// Successful creation means that no main process exists QCryptographicHash appData(QCryptographicHash::Sha256);
// So we start a QLocalServer to listen for connections appData.addData("SingleApplication", 17);
QLocalServer::removeServer( blockServerName ); appData.addData(SingleApplication::app_t::applicationName().toUtf8());
server = new QLocalServer(); appData.addData(SingleApplication::app_t::organizationName().toUtf8());
appData.addData(SingleApplication::app_t::organizationDomain().toUtf8());
// Restrict access to the socket according to the if (!(options & SingleApplication::Mode::ExcludeAppVersion)) {
// SingleApplication::Mode::User flag on User level or no restrictions appData.addData(SingleApplication::app_t::applicationVersion().toUtf8());
if( options & SingleApplication::Mode::User ) { }
server->setSocketOptions( QLocalServer::UserAccessOption );
} else {
server->setSocketOptions( QLocalServer::WorldAccessOption );
}
server->listen( blockServerName ); if (! (options & SingleApplication::Mode::ExcludeAppPath)) {
QObject::connect( #ifdef Q_OS_WIN
server, appData.addData(SingleApplication::app_t::applicationFilePath().toLower().toUtf8());
&QLocalServer::newConnection, #else
this, appData.addData(SingleApplication::app_t::applicationFilePath().toUtf8());
&SingleApplicationPrivate::slotConnectionEstablished
);
// Reset the number of connections
InstancesInfo* inst = static_cast <InstancesInfo*>( memory->data() );
inst->primary = true;
inst->primaryPid = q->applicationPid();
inst->checksum = blockChecksum();
instanceNumber = 0;
}
void SingleApplicationPrivate::startSecondary()
{
}
void SingleApplicationPrivate::connectToPrimary( int msecs, ConnectionType connectionType )
{
// Connect to the Local Server of the Primary Instance if not already
// connected.
if( socket == nullptr ) {
socket = new QLocalSocket();
}
// If already connected - we are done;
if( socket->state() == QLocalSocket::ConnectedState )
return;
// If not connect
if( socket->state() == QLocalSocket::UnconnectedState ||
socket->state() == QLocalSocket::ClosingState ) {
socket->connectToServer( blockServerName );
}
// Wait for being connected
if( socket->state() == QLocalSocket::ConnectingState ) {
socket->waitForConnected( msecs );
}
// Initialisation message according to the SingleApplication protocol
if( socket->state() == QLocalSocket::ConnectedState ) {
// Notify the parent that a new instance had been started;
QByteArray initMsg;
QDataStream writeStream(&initMsg, QIODevice::WriteOnly);
#if (QT_VERSION >= QT_VERSION_CHECK(5, 6, 0))
writeStream.setVersion(QDataStream::Qt_5_6);
#endif #endif
}
writeStream << blockServerName.toLatin1(); // User level block requires a user specific data in the hash
writeStream << static_cast<quint8>(connectionType); if (options & SingleApplication::Mode::User) {
writeStream << instanceNumber; #ifdef Q_OS_UNIX
quint16 checksum = qChecksum(initMsg.constData(), static_cast<quint32>(initMsg.length())); QByteArray username;
writeStream << checksum; #if defined(HAVE_GETEUID) && defined(HAVE_GETPWUID)
struct passwd *pw = getpwuid(geteuid());
// The header indicates the message length that follows if (pw) {
QByteArray header; username = pw->pw_name;
QDataStream headerStream(&header, QIODevice::WriteOnly);
#if (QT_VERSION >= QT_VERSION_CHECK(5, 6, 0))
headerStream.setVersion(QDataStream::Qt_5_6);
#endif
headerStream << static_cast <quint64>( initMsg.length() );
socket->write( header );
socket->write( initMsg );
socket->flush();
socket->waitForBytesWritten( msecs );
} }
#endif
if (username.isEmpty()) username = qgetenv("USER");
appData.addData(username);
#endif
#ifdef Q_OS_WIN
wchar_t username [ UNLEN + 1 ];
// Specifies size of the buffer on input
DWORD usernameLength = UNLEN + 1;
if (GetUserNameW(username, &usernameLength)) {
appData.addData(QString::fromWCharArray(username).toUtf8());
}
else {
appData.addData(qgetenv("USERNAME"));
}
#endif
}
// Replace the backslash in RFC 2045 Base64 [a-zA-Z0-9+/=] to comply with server naming requirements.
blockServerName = appData.result().toBase64().replace("/", "_");
} }
quint16 SingleApplicationPrivate::blockChecksum() void SingleApplicationPrivate::initializeMemoryBlock() {
{
return qChecksum( InstancesInfo* inst = static_cast<InstancesInfo*>(memory->data());
static_cast <const char *>( memory->data() ), inst->primary = false;
offsetof( InstancesInfo, checksum ) inst->secondary = 0;
); inst->primaryPid = -1;
inst->checksum = blockChecksum();
} }
qint64 SingleApplicationPrivate::primaryPid() void SingleApplicationPrivate::startPrimary() {
{
qint64 pid;
memory->lock(); Q_Q(SingleApplication);
InstancesInfo* inst = static_cast<InstancesInfo*>( memory->data() );
pid = inst->primaryPid; // Successful creation means that no main process exists
memory->unlock(); // So we start a QLocalServer to listen for connections
QLocalServer::removeServer(blockServerName);
server = new QLocalServer();
// Restrict access to the socket according to the
// SingleApplication::Mode::User flag on User level or no restrictions
if (options & SingleApplication::Mode::User) {
server->setSocketOptions(QLocalServer::UserAccessOption);
}
else {
server->setSocketOptions(QLocalServer::WorldAccessOption);
}
server->listen(blockServerName);
QObject::connect(server, &QLocalServer::newConnection, this, &SingleApplicationPrivate::slotConnectionEstablished);
// Reset the number of connections
InstancesInfo* inst = static_cast <InstancesInfo*>(memory->data());
inst->primary = true;
inst->primaryPid = q->applicationPid();
inst->checksum = blockChecksum();
instanceNumber = 0;
}
void SingleApplicationPrivate::startSecondary() {}
void SingleApplicationPrivate::connectToPrimary(const int msecs, const ConnectionType connectionType) {
// Connect to the Local Server of the Primary Instance if not already connected.
if (socket == nullptr) {
socket = new QLocalSocket();
}
// If already connected - we are done;
if (socket->state() == QLocalSocket::ConnectedState)
return;
// If not connect
if (socket->state() == QLocalSocket::UnconnectedState ||
socket->state() == QLocalSocket::ClosingState) {
socket->connectToServer(blockServerName);
}
// Wait for being connected
if (socket->state() == QLocalSocket::ConnectingState) {
socket->waitForConnected(msecs);
}
// Initialisation message according to the SingleApplication protocol
if (socket->state() == QLocalSocket::ConnectedState) {
// Notify the parent that a new instance had been started;
QByteArray initMsg;
QDataStream writeStream(&initMsg, QIODevice::WriteOnly);
writeStream.setVersion(QDataStream::Qt_5_6);
writeStream << blockServerName.toLatin1();
writeStream << static_cast<quint8>(connectionType);
writeStream << instanceNumber;
quint16 checksum = qChecksum(initMsg.constData(), static_cast<quint32>(initMsg.length()));
writeStream << checksum;
// The header indicates the message length that follows
QByteArray header;
QDataStream headerStream(&header, QIODevice::WriteOnly);
headerStream.setVersion(QDataStream::Qt_5_6);
headerStream << static_cast <quint64>(initMsg.length());
socket->write(header);
socket->write(initMsg);
socket->flush();
socket->waitForBytesWritten(msecs);
}
}
quint16 SingleApplicationPrivate::blockChecksum() {
return qChecksum(static_cast <const char *>(memory->data()), offsetof(InstancesInfo, checksum));
}
qint64 SingleApplicationPrivate::primaryPid() {
qint64 pid;
memory->lock();
InstancesInfo* inst = static_cast<InstancesInfo*>(memory->data());
pid = inst->primaryPid;
memory->unlock();
return pid;
return pid;
} }
/** /**
* @brief Executed when a connection has been made to the LocalServer * @brief Executed when a connection has been made to the LocalServer
*/ */
void SingleApplicationPrivate::slotConnectionEstablished() void SingleApplicationPrivate::slotConnectionEstablished() {
{
QLocalSocket *nextConnSocket = server->nextPendingConnection();
connectionMap.insert(nextConnSocket, ConnectionInfo());
QObject::connect(nextConnSocket, &QLocalSocket::aboutToClose, QLocalSocket *nextConnSocket = server->nextPendingConnection();
[nextConnSocket, this]() { connectionMap.insert(nextConnSocket, ConnectionInfo());
auto &info = connectionMap[nextConnSocket];
Q_EMIT this->slotClientConnectionClosed( nextConnSocket, info.instanceId );
}
);
QObject::connect(nextConnSocket, &QLocalSocket::disconnected, QObject::connect(nextConnSocket, &QLocalSocket::aboutToClose,
[nextConnSocket, this](){ [nextConnSocket, this]() {
connectionMap.remove(nextConnSocket); auto &info = connectionMap[nextConnSocket];
nextConnSocket->deleteLater(); Q_EMIT this->slotClientConnectionClosed(nextConnSocket, info.instanceId);
} }
); );
QObject::connect(nextConnSocket, &QLocalSocket::disconnected,
[nextConnSocket, this](){
connectionMap.remove(nextConnSocket);
nextConnSocket->deleteLater();
}
);
QObject::connect(nextConnSocket, &QLocalSocket::readyRead,
[nextConnSocket, this]() {
auto &info = connectionMap[nextConnSocket];
switch(info.stage) {
case StageHeader:
readInitMessageHeader(nextConnSocket);
break;
case StageBody:
readInitMessageBody(nextConnSocket);
break;
case StageConnected:
Q_EMIT this->slotDataAvailable(nextConnSocket, info.instanceId);
break;
default:
break;
};
}
);
QObject::connect(nextConnSocket, &QLocalSocket::readyRead,
[nextConnSocket, this]() {
auto &info = connectionMap[nextConnSocket];
switch(info.stage) {
case StageHeader:
readInitMessageHeader(nextConnSocket);
break;
case StageBody:
readInitMessageBody(nextConnSocket);
break;
case StageConnected:
Q_EMIT this->slotDataAvailable( nextConnSocket, info.instanceId );
break;
default:
break;
};
}
);
} }
void SingleApplicationPrivate::readInitMessageHeader( QLocalSocket *sock ) void SingleApplicationPrivate::readInitMessageHeader(QLocalSocket *sock) {
{
if (!connectionMap.contains( sock )) {
return;
}
if( sock->bytesAvailable() < ( qint64 )sizeof( quint64 ) ) { if (!connectionMap.contains(sock)) {
return; return;
} }
QDataStream headerStream( sock ); if (sock->bytesAvailable() < static_cast<qint64>(sizeof(quint64))) {
return;
}
#if (QT_VERSION >= QT_VERSION_CHECK(5, 6, 0)) QDataStream headerStream(sock);
headerStream.setVersion( QDataStream::Qt_5_6 );
#endif
// Read the header to know the message length headerStream.setVersion(QDataStream::Qt_5_6);
quint64 msgLen = 0;
headerStream >> msgLen; // Read the header to know the message length
ConnectionInfo &info = connectionMap[sock]; quint64 msgLen = 0;
info.stage = StageBody; headerStream >> msgLen;
info.msgLen = msgLen; ConnectionInfo &info = connectionMap[sock];
info.stage = StageBody;
info.msgLen = msgLen;
if (sock->bytesAvailable() >= static_cast<qint64>(msgLen)) {
readInitMessageBody(sock);
}
if ( sock->bytesAvailable() >= (qint64) msgLen ) {
readInitMessageBody( sock );
}
} }
void SingleApplicationPrivate::readInitMessageBody( QLocalSocket *sock ) void SingleApplicationPrivate::readInitMessageBody(QLocalSocket *sock) {
{
Q_Q(SingleApplication);
if (!connectionMap.contains( sock )) { Q_Q(SingleApplication);
return;
}
ConnectionInfo &info = connectionMap[sock]; if (!connectionMap.contains(sock)) {
if( sock->bytesAvailable() < ( qint64 )info.msgLen ) { return;
return; }
}
// Read the message body ConnectionInfo &info = connectionMap[sock];
QByteArray msgBytes = sock->read(info.msgLen); if (sock->bytesAvailable() < static_cast<qint64>(info.msgLen)) {
QDataStream readStream(msgBytes); return;
}
#if (QT_VERSION >= QT_VERSION_CHECK(5, 6, 0)) // Read the message body
readStream.setVersion( QDataStream::Qt_5_6 ); QByteArray msgBytes = sock->read(info.msgLen);
#endif QDataStream readStream(msgBytes);
// server name readStream.setVersion(QDataStream::Qt_5_6);
QByteArray latin1Name;
readStream >> latin1Name;
// connection type // server name
ConnectionType connectionType = InvalidConnection; QByteArray latin1Name;
quint8 connTypeVal = InvalidConnection; readStream >> latin1Name;
readStream >> connTypeVal;
connectionType = static_cast <ConnectionType>( connTypeVal );
// instance id // connection type
quint32 instanceId = 0; ConnectionType connectionType = InvalidConnection;
readStream >> instanceId; quint8 connTypeVal = InvalidConnection;
readStream >> connTypeVal;
connectionType = static_cast <ConnectionType>(connTypeVal);
// checksum // instance id
quint16 msgChecksum = 0; quint32 instanceId = 0;
readStream >> msgChecksum; readStream >> instanceId;
const quint16 actualChecksum = qChecksum( msgBytes.constData(), static_cast<quint32>( msgBytes.length() - sizeof( quint16 ) ) ); // checksum
quint16 msgChecksum = 0;
readStream >> msgChecksum;
bool isValid = readStream.status() == QDataStream::Ok && const quint16 actualChecksum = qChecksum(msgBytes.constData(), static_cast<quint32>(msgBytes.length() - sizeof(quint16)));
QLatin1String(latin1Name) == blockServerName &&
msgChecksum == actualChecksum;
if( !isValid ) { bool isValid = readStream.status() == QDataStream::Ok && QLatin1String(latin1Name) == blockServerName && msgChecksum == actualChecksum;
sock->close();
return;
}
info.instanceId = instanceId; if (!isValid) {
info.stage = StageConnected; sock->close();
return;
}
if( connectionType == NewInstance || info.instanceId = instanceId;
( connectionType == SecondaryInstance && info.stage = StageConnected;
options & SingleApplication::Mode::SecondaryNotification ) )
{ if (connectionType == NewInstance || (connectionType == SecondaryInstance && options & SingleApplication::Mode::SecondaryNotification)) {
Q_EMIT q->instanceStarted(); Q_EMIT q->instanceStarted();
} }
if (sock->bytesAvailable() > 0) {
Q_EMIT this->slotDataAvailable(sock, instanceId);
}
if (sock->bytesAvailable() > 0) {
Q_EMIT this->slotDataAvailable( sock, instanceId );
}
} }
void SingleApplicationPrivate::slotDataAvailable( QLocalSocket *dataSocket, quint32 instanceId ) void SingleApplicationPrivate::slotDataAvailable(QLocalSocket *dataSocket, const quint32 instanceId) {
{
Q_Q(SingleApplication); Q_Q(SingleApplication);
Q_EMIT q->receivedMessage( instanceId, dataSocket->readAll() ); Q_EMIT q->receivedMessage(instanceId, dataSocket->readAll());
} }
void SingleApplicationPrivate::slotClientConnectionClosed( QLocalSocket *closedSocket, quint32 instanceId ) void SingleApplicationPrivate::slotClientConnectionClosed(QLocalSocket *closedSocket, const quint32 instanceId) {
{
if( closedSocket->bytesAvailable() > 0 ) if (closedSocket->bytesAvailable() > 0)
Q_EMIT slotDataAvailable( closedSocket, instanceId ); Q_EMIT slotDataAvailable(closedSocket, instanceId);
} }

View File

@@ -24,76 +24,82 @@
// W A R N I N G !!! // W A R N I N G !!!
// ----------------- // -----------------
// //
// This file is not part of the SingleApplication API. It is used purely as an // This is a modified version of SingleApplication,
// implementation detail. This header file may change from version to // The original version is at:
// version without notice, or may even be removed. //
// https://github.com/itay-grudev/SingleApplication
//
// //
#ifndef SINGLEAPPLICATION_P_H #ifndef SINGLEAPPLICATION_P_H
#define SINGLEAPPLICATION_P_H #define SINGLEAPPLICATION_P_H
#include <QtCore/QSharedMemory> #include <QtGlobal>
#include <QtNetwork/QLocalServer> #include <QObject>
#include <QtNetwork/QLocalSocket> #include <QMap>
#include "singleapplication.h" #include "singleapplication.h"
class QLocalServer;
class QLocalSocket;
class QSharedMemory;
struct InstancesInfo { struct InstancesInfo {
bool primary; bool primary;
quint32 secondary; quint32 secondary;
qint64 primaryPid; qint64 primaryPid;
quint16 checksum; quint16 checksum;
}; };
struct ConnectionInfo { struct ConnectionInfo {
explicit ConnectionInfo() : explicit ConnectionInfo() : msgLen(0), instanceId(0), stage(0) {}
msgLen(0), instanceId(0), stage(0) {} qint64 msgLen;
qint64 msgLen; quint32 instanceId;
quint32 instanceId; quint8 stage;
quint8 stage;
}; };
class SingleApplicationPrivate : public QObject { class SingleApplicationPrivate : public QObject {
Q_OBJECT Q_OBJECT
public: public:
enum ConnectionType : quint8 { enum ConnectionType : quint8 {
InvalidConnection = 0, InvalidConnection = 0,
NewInstance = 1, NewInstance = 1,
SecondaryInstance = 2, SecondaryInstance = 2,
Reconnect = 3 Reconnect = 3
}; };
enum ConnectionStage : quint8 { enum ConnectionStage : quint8 {
StageHeader = 0, StageHeader = 0,
StageBody = 1, StageBody = 1,
StageConnected = 2, StageConnected = 2,
}; };
Q_DECLARE_PUBLIC(SingleApplication) Q_DECLARE_PUBLIC(SingleApplication)
SingleApplicationPrivate( SingleApplication *q_ptr ); explicit SingleApplicationPrivate(SingleApplication *_q_ptr);
~SingleApplicationPrivate(); ~SingleApplicationPrivate() override;
void genBlockServerName(); void genBlockServerName();
void initializeMemoryBlock(); void initializeMemoryBlock();
void startPrimary(); void startPrimary();
void startSecondary(); void startSecondary();
void connectToPrimary(int msecs, ConnectionType connectionType ); void connectToPrimary(const int msecs, const ConnectionType connectionType);
quint16 blockChecksum(); quint16 blockChecksum();
qint64 primaryPid(); qint64 primaryPid();
void readInitMessageHeader(QLocalSocket *socket); void readInitMessageHeader(QLocalSocket *socket);
void readInitMessageBody(QLocalSocket *socket); void readInitMessageBody(QLocalSocket *socket);
SingleApplication *q_ptr; SingleApplication *q_ptr;
QSharedMemory *memory; QSharedMemory *memory;
QLocalSocket *socket; QLocalSocket *socket;
QLocalServer *server; QLocalServer *server;
quint32 instanceNumber; quint32 instanceNumber;
QString blockServerName; QString blockServerName;
SingleApplication::Options options; SingleApplication::Options options;
QMap<QLocalSocket*, ConnectionInfo> connectionMap; QMap<QLocalSocket*, ConnectionInfo> connectionMap;
public Q_SLOTS: public slots:
void slotConnectionEstablished(); void slotConnectionEstablished();
void slotDataAvailable( QLocalSocket*, quint32 ); void slotDataAvailable(QLocalSocket*, const quint32);
void slotClientConnectionClosed( QLocalSocket*, quint32 ); void slotClientConnectionClosed(QLocalSocket*, const quint32);
}; };
#endif // SINGLEAPPLICATION_P_H #endif // SINGLEAPPLICATION_P_H

View File

@@ -20,12 +20,33 @@
// OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN // OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN
// THE SOFTWARE. // THE SOFTWARE.
//
// W A R N I N G !!!
// -----------------
//
// This is a modified version of SingleApplication,
// The original version is at:
//
// https://github.com/itay-grudev/SingleApplication
//
//
#include <cstdlib>
#include <limits>
#include <QtGlobal>
#include <QCoreApplication> #include <QCoreApplication>
#include <QtCore/QTime> #include <QThread>
#include <QtCore/QThread> #include <QSharedMemory>
#include <QtCore/QDateTime> #include <QLocalSocket>
#include <QtCore/QByteArray> #include <QByteArray>
#include <QtCore/QSharedMemory> #include <QElapsedTimer>
#include <QtDebug>
#if QT_VERSION >= QT_VERSION_CHECK(5, 10, 0)
# include <QRandomGenerator>
#else
# include <QDateTime>
#endif
#include "singlecoreapplication.h" #include "singlecoreapplication.h"
#include "singlecoreapplication_p.h" #include "singlecoreapplication_p.h"
@@ -37,139 +58,140 @@
* @param argv * @param argv
* @param {bool} allowSecondaryInstances * @param {bool} allowSecondaryInstances
*/ */
SingleCoreApplication::SingleCoreApplication( int &argc, char *argv[], bool allowSecondary, Options options, int timeout ) SingleCoreApplication::SingleCoreApplication(int &argc, char *argv[], bool allowSecondary, Options options, int timeout)
: app_t( argc, argv ), d_ptr( new SingleCoreApplicationPrivate( this ) ) : app_t(argc, argv), d_ptr(new SingleCoreApplicationPrivate(this)) {
{
Q_D(SingleCoreApplication);
// Store the current mode of the program Q_D(SingleCoreApplication);
d->options = options;
// Generating an application ID used for identifying the shared memory // Store the current mode of the program
// block and QLocalServer d->options = options;
d->genBlockServerName();
// Generating an application ID used for identifying the shared memory block and QLocalServer
d->genBlockServerName();
#ifdef Q_OS_UNIX #ifdef Q_OS_UNIX
// By explicitly attaching it and then deleting it we make sure that the // By explicitly attaching it and then deleting it we make sure that the
// memory is deleted even after the process has crashed on Unix. // memory is deleted even after the process has crashed on Unix.
d->memory = new QSharedMemory( d->blockServerName ); d->memory = new QSharedMemory(d->blockServerName);
d->memory->attach(); d->memory->attach();
delete d->memory; delete d->memory;
#endif #endif
// Guarantee thread safe behaviour with a shared memory block. // Guarantee thread safe behaviour with a shared memory block.
d->memory = new QSharedMemory( d->blockServerName ); d->memory = new QSharedMemory(d->blockServerName);
// Create a shared memory block // Create a shared memory block
if( d->memory->create( sizeof( InstancesInfo ) ) ) { if (d->memory->create(sizeof(InstancesInfo))) {
// Initialize the shared memory block // Initialize the shared memory block
d->memory->lock(); d->memory->lock();
d->initializeMemoryBlock(); d->initializeMemoryBlock();
d->memory->unlock(); d->memory->unlock();
} else { }
// Attempt to attach to the memory segment else {
if( ! d->memory->attach() ) { // Attempt to attach to the memory segment
qCritical() << "SingleCoreApplication: Unable to attach to shared memory block."; if (!d->memory->attach()) {
qCritical() << d->memory->errorString(); qCritical() << "SingleCoreApplication: Unable to attach to shared memory block.";
delete d; qCritical() << d->memory->errorString();
::exit( EXIT_FAILURE ); delete d;
} ::exit(EXIT_FAILURE);
} }
}
InstancesInfo* inst = static_cast<InstancesInfo*>( d->memory->data() ); InstancesInfo* inst = static_cast<InstancesInfo*>(d->memory->data());
QTime time; QElapsedTimer time;
time.start(); time.start();
// Make sure the shared memory block is initialised and in consistent state // Make sure the shared memory block is initialised and in consistent state
while( true ) { while (true) {
d->memory->lock(); d->memory->lock();
if( d->blockChecksum() == inst->checksum ) break; if(d->blockChecksum() == inst->checksum) break;
if( time.elapsed() > 5000 ) { if (time.elapsed() > 5000) {
qWarning() << "SingleCoreApplication: Shared memory block has been in an inconsistent state from more than 5s. Assuming primary instance failure."; qWarning() << "SingleCoreApplication: Shared memory block has been in an inconsistent state from more than 5s. Assuming primary instance failure.";
d->initializeMemoryBlock(); d->initializeMemoryBlock();
}
d->memory->unlock();
// Random sleep here limits the probability of a collision between two racing apps
qsrand( QDateTime::currentMSecsSinceEpoch() % std::numeric_limits<uint>::max() );
QThread::sleep( 8 + static_cast <unsigned long>( static_cast <float>( qrand() ) / RAND_MAX * 10 ) );
}
if( inst->primary == false) {
d->startPrimary();
d->memory->unlock();
return;
}
// Check if another instance can be started
if( allowSecondary ) {
inst->secondary += 1;
inst->checksum = d->blockChecksum();
d->instanceNumber = inst->secondary;
d->startSecondary();
if( d->options & Mode::SecondaryNotification ) {
d->connectToPrimary( timeout, SingleCoreApplicationPrivate::SecondaryInstance );
}
d->memory->unlock();
return;
} }
d->memory->unlock(); d->memory->unlock();
d->connectToPrimary( timeout, SingleCoreApplicationPrivate::NewInstance ); // Random sleep here limits the probability of a collision between two racing apps
#if QT_VERSION >= QT_VERSION_CHECK(5, 10, 0)
QThread::sleep(QRandomGenerator::global()->bounded(8u, 18u));
#else
qsrand(QDateTime::currentMSecsSinceEpoch() % std::numeric_limits<uint>::max());
QThread::sleep(8 + static_cast<unsigned long>(static_cast <float>(qrand()) / RAND_MAX * 10));
#endif
}
delete d; if (inst->primary == false) {
d->startPrimary();
d->memory->unlock();
return;
}
// Check if another instance can be started
if (allowSecondary) {
inst->secondary += 1;
inst->checksum = d->blockChecksum();
d->instanceNumber = inst->secondary;
d->startSecondary();
if(d->options & Mode::SecondaryNotification) {
d->connectToPrimary(timeout, SingleCoreApplicationPrivate::SecondaryInstance);
}
d->memory->unlock();
return;
}
d->memory->unlock();
d->connectToPrimary(timeout, SingleCoreApplicationPrivate::NewInstance);
delete d;
::exit(EXIT_SUCCESS);
::exit( EXIT_SUCCESS );
} }
/** /**
* @brief Destructor * @brief Destructor
*/ */
SingleCoreApplication::~SingleCoreApplication() SingleCoreApplication::~SingleCoreApplication() {
{ Q_D(SingleCoreApplication);
Q_D(SingleCoreApplication); delete d;
delete d;
} }
bool SingleCoreApplication::isPrimary() bool SingleCoreApplication::isPrimary() {
{ Q_D(SingleCoreApplication);
Q_D(SingleCoreApplication); return d->server != nullptr;
return d->server != nullptr;
} }
bool SingleCoreApplication::isSecondary() bool SingleCoreApplication::isSecondary() {
{ Q_D(SingleCoreApplication);
Q_D(SingleCoreApplication); return d->server == nullptr;
return d->server == nullptr;
} }
quint32 SingleCoreApplication::instanceId() quint32 SingleCoreApplication::instanceId() {
{ Q_D(SingleCoreApplication);
Q_D(SingleCoreApplication); return d->instanceNumber;
return d->instanceNumber;
} }
qint64 SingleCoreApplication::primaryPid() qint64 SingleCoreApplication::primaryPid() {
{ Q_D(SingleCoreApplication);
Q_D(SingleCoreApplication); return d->primaryPid();
return d->primaryPid();
} }
bool SingleCoreApplication::sendMessage( QByteArray message, int timeout ) bool SingleCoreApplication::sendMessage(QByteArray message, int timeout) {
{
Q_D(SingleCoreApplication);
// Nobody to connect to Q_D(SingleCoreApplication);
if( isPrimary() ) return false;
// Make sure the socket is connected // Nobody to connect to
d->connectToPrimary( timeout, SingleCoreApplicationPrivate::Reconnect ); if(isPrimary()) return false;
// Make sure the socket is connected
d->connectToPrimary(timeout, SingleCoreApplicationPrivate::Reconnect);
d->socket->write(message);
bool dataWritten = d->socket->waitForBytesWritten(timeout);
d->socket->flush();
return dataWritten;
d->socket->write( message );
bool dataWritten = d->socket->flush();
d->socket->waitForBytesWritten( timeout );
return dataWritten;
} }

View File

@@ -20,109 +20,120 @@
// OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN // OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN
// THE SOFTWARE. // THE SOFTWARE.
//
// W A R N I N G !!!
// -----------------
//
// This is a modified version of SingleApplication,
// The original version is at:
//
// https://github.com/itay-grudev/SingleApplication
//
//
#ifndef SINGLECOREAPPLICATION_H #ifndef SINGLECOREAPPLICATION_H
#define SINGLECOREAPPLICATION_H #define SINGLECOREAPPLICATION_H
#include <QtCore/QtGlobal> #include <QtGlobal>
#include <QCoreApplication> #include <QCoreApplication>
#include <QtNetwork/QLocalSocket> #include <QFlags>
#include <QByteArray>
class SingleCoreApplicationPrivate; class SingleCoreApplicationPrivate;
/** /**
* @brief The SingleCoreApplication class handles multipe instances of the same * @brief The SingleCoreApplication class handles multiple instances of the same
* Application * Application
* @see QCoreApplication * @see QCoreApplication
*/ */
class SingleCoreApplication : public QCoreApplication class SingleCoreApplication : public QCoreApplication {
{ Q_OBJECT
Q_OBJECT
typedef QCoreApplication app_t; typedef QCoreApplication app_t;
public: public:
/** /**
* @brief Mode of operation of SingleCoreApplication. * @brief Mode of operation of SingleCoreApplication.
* Whether the block should be user-wide or system-wide and whether the * Whether the block should be user-wide or system-wide and whether the
* primary instance should be notified when a secondary instance had been * primary instance should be notified when a secondary instance had been
* started. * started.
* @note Operating system can restrict the shared memory blocks to the same * @note Operating system can restrict the shared memory blocks to the same
* user, in which case the User/System modes will have no effect and the * user, in which case the User/System modes will have no effect and the
* block will be user wide. * block will be user wide.
* @enum * @enum
*/ */
enum Mode { enum Mode {
User = 1 << 0, User = 1 << 0,
System = 1 << 1, System = 1 << 1,
SecondaryNotification = 1 << 2, SecondaryNotification = 1 << 2,
ExcludeAppVersion = 1 << 3, ExcludeAppVersion = 1 << 3,
ExcludeAppPath = 1 << 4 ExcludeAppPath = 1 << 4
}; };
Q_DECLARE_FLAGS(Options, Mode) Q_DECLARE_FLAGS(Options, Mode)
/** /**
* @brief Intitializes a SingleCoreApplication instance with argc command line * @brief Intitializes a SingleCoreApplication instance with argc command line
* arguments in argv * arguments in argv
* @arg {int &} argc - Number of arguments in argv * @arg {int &} argc - Number of arguments in argv
* @arg {const char *[]} argv - Supplied command line arguments * @arg {const char *[]} argv - Supplied command line arguments
* @arg {bool} allowSecondary - Whether to start the instance as secondary * @arg {bool} allowSecondary - Whether to start the instance as secondary
* if there is already a primary instance. * if there is already a primary instance.
* @arg {Mode} mode - Whether for the SingleCoreApplication block to be applied * @arg {Mode} mode - Whether for the SingleCoreApplication block to be applied
* User wide or System wide. * User wide or System wide.
* @arg {int} timeout - Timeout to wait in miliseconds. * @arg {int} timeout - Timeout to wait in milliseconds.
* @note argc and argv may be changed as Qt removes arguments that it * @note argc and argv may be changed as Qt removes arguments that it
* recognizes * recognizes
* @note Mode::SecondaryNotification only works if set on both the primary * @note Mode::SecondaryNotification only works if set on both the primary
* instance and the secondary instance. * instance and the secondary instance.
* @note The timeout is just a hint for the maximum time of blocking * @note The timeout is just a hint for the maximum time of blocking
* operations. It does not guarantee that the SingleCoreApplication * operations. It does not guarantee that the SingleCoreApplication
* initialisation will be completed in given time, though is a good hint. * initialisation will be completed in given time, though is a good hint.
* Usually 4*timeout would be the worst case (fail) scenario. * Usually 4*timeout would be the worst case (fail) scenario.
* @see See the corresponding QAPPLICATION_CLASS constructor for reference * @see See the corresponding QAPPLICATION_CLASS constructor for reference
*/ */
explicit SingleCoreApplication( int &argc, char *argv[], bool allowSecondary = false, Options options = Mode::User, int timeout = 1000 ); explicit SingleCoreApplication( int &argc, char *argv[], bool allowSecondary = false, Options options = Mode::User, int timeout = 1000 );
~SingleCoreApplication(); ~SingleCoreApplication() override;
/** /**
* @brief Returns if the instance is the primary instance * @brief Returns if the instance is the primary instance
* @returns {bool} * @returns {bool}
*/ */
bool isPrimary(); bool isPrimary();
/** /**
* @brief Returns if the instance is a secondary instance * @brief Returns if the instance is a secondary instance
* @returns {bool} * @returns {bool}
*/ */
bool isSecondary(); bool isSecondary();
/** /**
* @brief Returns a unique identifier for the current instance * @brief Returns a unique identifier for the current instance
* @returns {qint32} * @returns {qint32}
*/ */
quint32 instanceId(); quint32 instanceId();
/** /**
* @brief Returns the process ID (PID) of the primary instance * @brief Returns the process ID (PID) of the primary instance
* @returns {qint64} * @returns {qint64}
*/ */
qint64 primaryPid(); qint64 primaryPid();
/** /**
* @brief Sends a message to the primary instance. Returns true on success. * @brief Sends a message to the primary instance. Returns true on success.
* @param {int} timeout - Timeout for connecting * @param {int} timeout - Timeout for connecting
* @returns {bool} * @returns {bool}
* @note sendMessage() will return false if invoked from the primary * @note sendMessage() will return false if invoked from the primary
* instance. * instance.
*/ */
bool sendMessage( QByteArray message, int timeout = 100 ); bool sendMessage( QByteArray message, int timeout = 1000 );
Q_SIGNALS: signals:
void instanceStarted(); void instanceStarted();
void receivedMessage( quint32 instanceId, QByteArray message ); void receivedMessage( quint32 instanceId, QByteArray message );
private: private:
SingleCoreApplicationPrivate *d_ptr; SingleCoreApplicationPrivate *d_ptr;
Q_DECLARE_PRIVATE(SingleCoreApplication) Q_DECLARE_PRIVATE(SingleCoreApplication)
}; };
Q_DECLARE_OPERATORS_FOR_FLAGS(SingleCoreApplication::Options) Q_DECLARE_OPERATORS_FOR_FLAGS(SingleCoreApplication::Options)

View File

@@ -24,381 +24,378 @@
// W A R N I N G !!! // W A R N I N G !!!
// ----------------- // -----------------
// //
// This file is not part of the SingleCoreApplication API. It is used purely as an // This is a modified version of SingleApplication,
// implementation detail. This header file may change from version to // The original version is at:
// version without notice, or may even be removed.
// //
// https://github.com/itay-grudev/SingleApplication
//
//
#include "config.h"
#include <QtGlobal>
#include <cstdlib> #include <cstdlib>
#include <cstddef> #include <cstddef>
#include <QtCore/QDir> #ifdef Q_OS_UNIX
#include <QtCore/QProcess> # include <unistd.h>
#include <QtCore/QByteArray> # include <sys/types.h>
#include <QtCore/QSemaphore> # include <pwd.h>
#include <QtCore/QDataStream> #endif
#include <QtCore/QStandardPaths>
#include <QtCore/QCryptographicHash> #include <QIODevice>
#include <QtNetwork/QLocalServer> #include <QSharedMemory>
#include <QtNetwork/QLocalSocket> #include <QByteArray>
#include <QDataStream>
#include <QCryptographicHash>
#include <QLocalServer>
#include <QLocalSocket>
#include <QDir>
#include "singlecoreapplication.h" #include "singlecoreapplication.h"
#include "singlecoreapplication_p.h" #include "singlecoreapplication_p.h"
#ifdef Q_OS_WIN #ifdef Q_OS_WIN
#include <windows.h> # include <windows.h>
#include <lmcons.h> # include <lmcons.h>
#endif #endif
SingleCoreApplicationPrivate::SingleCoreApplicationPrivate( SingleCoreApplication *q_ptr ) SingleCoreApplicationPrivate::SingleCoreApplicationPrivate(SingleCoreApplication *_q_ptr)
: q_ptr( q_ptr ) : q_ptr(_q_ptr),
{ memory(nullptr),
server = nullptr; socket(nullptr),
socket = nullptr; server(nullptr),
memory = nullptr; instanceNumber(-1)
instanceNumber = -1; {}
}
SingleCoreApplicationPrivate::~SingleCoreApplicationPrivate() SingleCoreApplicationPrivate::~SingleCoreApplicationPrivate() {
{
if( socket != nullptr ) {
socket->close();
delete socket;
}
memory->lock(); if (socket != nullptr) {
InstancesInfo* inst = static_cast<InstancesInfo*>(memory->data()); socket->close();
if( server != nullptr ) { delete socket;
server->close(); }
delete server;
inst->primary = false;
inst->primaryPid = -1;
inst->checksum = blockChecksum();
}
memory->unlock();
delete memory; memory->lock();
} InstancesInfo* inst = static_cast<InstancesInfo*>(memory->data());
if (server != nullptr) {
void SingleCoreApplicationPrivate::genBlockServerName() server->close();
{ delete server;
QCryptographicHash appData( QCryptographicHash::Sha256 );
appData.addData( "SingleApplication", 17 );
appData.addData( SingleCoreApplication::app_t::applicationName().toUtf8() );
appData.addData( SingleCoreApplication::app_t::organizationName().toUtf8() );
appData.addData( SingleCoreApplication::app_t::organizationDomain().toUtf8() );
if( ! (options & SingleCoreApplication::Mode::ExcludeAppVersion) ) {
appData.addData( SingleCoreApplication::app_t::applicationVersion().toUtf8() );
}
if( ! (options & SingleCoreApplication::Mode::ExcludeAppPath) ) {
#ifdef Q_OS_WIN
appData.addData( SingleCoreApplication::app_t::applicationFilePath().toLower().toUtf8() );
#else
appData.addData( SingleCoreApplication::app_t::applicationFilePath().toUtf8() );
#endif
}
// User level block requires a user specific data in the hash
if( options & SingleCoreApplication::Mode::User ) {
#ifdef Q_OS_WIN
wchar_t username [ UNLEN + 1 ];
// Specifies size of the buffer on input
DWORD usernameLength = UNLEN + 1;
if( GetUserNameW( username, &usernameLength ) ) {
appData.addData( QString::fromWCharArray(username).toUtf8() );
} else {
appData.addData( QStandardPaths::standardLocations( QStandardPaths::HomeLocation ).join("").toUtf8() );
}
#endif
#ifdef Q_OS_UNIX
QProcess process;
process.start( "whoami" );
if( process.waitForFinished( 100 ) &&
process.exitCode() == QProcess::NormalExit) {
appData.addData( process.readLine() );
} else {
appData.addData(
QDir(
QStandardPaths::standardLocations( QStandardPaths::HomeLocation ).first()
).absolutePath().toUtf8()
);
}
#endif
}
// Replace the backslash in RFC 2045 Base64 [a-zA-Z0-9+/=] to comply with
// server naming requirements.
blockServerName = appData.result().toBase64().replace("/", "_");
}
void SingleCoreApplicationPrivate::initializeMemoryBlock()
{
InstancesInfo* inst = static_cast<InstancesInfo*>( memory->data() );
inst->primary = false; inst->primary = false;
inst->secondary = 0;
inst->primaryPid = -1; inst->primaryPid = -1;
inst->checksum = blockChecksum(); inst->checksum = blockChecksum();
}
memory->unlock();
delete memory;
} }
void SingleCoreApplicationPrivate::startPrimary() void SingleCoreApplicationPrivate::genBlockServerName() {
{
Q_Q(SingleCoreApplication);
// Successful creation means that no main process exists QCryptographicHash appData(QCryptographicHash::Sha256);
// So we start a QLocalServer to listen for connections appData.addData("SingleApplication", 17);
QLocalServer::removeServer( blockServerName ); appData.addData(SingleCoreApplication::app_t::applicationName().toUtf8());
server = new QLocalServer(); appData.addData(SingleCoreApplication::app_t::organizationName().toUtf8());
appData.addData(SingleCoreApplication::app_t::organizationDomain().toUtf8());
// Restrict access to the socket according to the if (!(options & SingleCoreApplication::Mode::ExcludeAppVersion)) {
// SingleCoreApplication::Mode::User flag on User level or no restrictions appData.addData(SingleCoreApplication::app_t::applicationVersion().toUtf8());
if( options & SingleCoreApplication::Mode::User ) { }
server->setSocketOptions( QLocalServer::UserAccessOption );
} else {
server->setSocketOptions( QLocalServer::WorldAccessOption );
}
server->listen( blockServerName ); if (!(options & SingleCoreApplication::Mode::ExcludeAppPath)) {
QObject::connect( #ifdef Q_OS_WIN
server, appData.addData(SingleCoreApplication::app_t::applicationFilePath().toLower().toUtf8());
&QLocalServer::newConnection, #else
this, appData.addData(SingleCoreApplication::app_t::applicationFilePath().toUtf8());
&SingleCoreApplicationPrivate::slotConnectionEstablished
);
// Reset the number of connections
InstancesInfo* inst = static_cast <InstancesInfo*>( memory->data() );
inst->primary = true;
inst->primaryPid = q->applicationPid();
inst->checksum = blockChecksum();
instanceNumber = 0;
}
void SingleCoreApplicationPrivate::startSecondary()
{
}
void SingleCoreApplicationPrivate::connectToPrimary( int msecs, ConnectionType connectionType )
{
// Connect to the Local Server of the Primary Instance if not already
// connected.
if( socket == nullptr ) {
socket = new QLocalSocket();
}
// If already connected - we are done;
if( socket->state() == QLocalSocket::ConnectedState )
return;
// If not connect
if( socket->state() == QLocalSocket::UnconnectedState ||
socket->state() == QLocalSocket::ClosingState ) {
socket->connectToServer( blockServerName );
}
// Wait for being connected
if( socket->state() == QLocalSocket::ConnectingState ) {
socket->waitForConnected( msecs );
}
// Initialisation message according to the SingleCoreApplication protocol
if( socket->state() == QLocalSocket::ConnectedState ) {
// Notify the parent that a new instance had been started;
QByteArray initMsg;
QDataStream writeStream(&initMsg, QIODevice::WriteOnly);
#if (QT_VERSION >= QT_VERSION_CHECK(5, 6, 0))
writeStream.setVersion(QDataStream::Qt_5_6);
#endif #endif
}
writeStream << blockServerName.toLatin1(); // User level block requires a user specific data in the hash
writeStream << static_cast<quint8>(connectionType); if (options & SingleCoreApplication::Mode::User) {
writeStream << instanceNumber; #ifdef Q_OS_UNIX
quint16 checksum = qChecksum(initMsg.constData(), static_cast<quint32>(initMsg.length())); QByteArray username;
writeStream << checksum; #if defined(HAVE_GETEUID) && defined(HAVE_GETPWUID)
struct passwd *pw = getpwuid(geteuid());
// The header indicates the message length that follows if (pw) {
QByteArray header; username = pw->pw_name;
QDataStream headerStream(&header, QIODevice::WriteOnly);
#if (QT_VERSION >= QT_VERSION_CHECK(5, 6, 0))
headerStream.setVersion(QDataStream::Qt_5_6);
#endif
headerStream << static_cast <quint64>( initMsg.length() );
socket->write( header );
socket->write( initMsg );
socket->flush();
socket->waitForBytesWritten( msecs );
} }
#endif
if (username.isEmpty()) username = qgetenv("USER");
appData.addData(username);
#endif
#ifdef Q_OS_WIN
wchar_t username [ UNLEN + 1 ];
// Specifies size of the buffer on input
DWORD usernameLength = UNLEN + 1;
if (GetUserNameW(username, &usernameLength)) {
appData.addData(QString::fromWCharArray(username).toUtf8());
}
else {
appData.addData(qgetenv("USERNAME"));
}
#endif
}
// Replace the backslash in RFC 2045 Base64 [a-zA-Z0-9+/=] to comply with server naming requirements.
blockServerName = appData.result().toBase64().replace("/", "_");
} }
quint16 SingleCoreApplicationPrivate::blockChecksum() void SingleCoreApplicationPrivate::initializeMemoryBlock() {
{
return qChecksum( InstancesInfo* inst = static_cast<InstancesInfo*>(memory->data());
static_cast <const char *>( memory->data() ), inst->primary = false;
offsetof( InstancesInfo, checksum ) inst->secondary = 0;
); inst->primaryPid = -1;
inst->checksum = blockChecksum();
} }
qint64 SingleCoreApplicationPrivate::primaryPid() void SingleCoreApplicationPrivate::startPrimary() {
{
qint64 pid;
memory->lock(); Q_Q(SingleCoreApplication);
InstancesInfo* inst = static_cast<InstancesInfo*>( memory->data() );
pid = inst->primaryPid; // Successful creation means that no main process exists
memory->unlock(); // So we start a QLocalServer to listen for connections
QLocalServer::removeServer(blockServerName);
server = new QLocalServer();
// Restrict access to the socket according to the
// SingleCoreApplication::Mode::User flag on User level or no restrictions
if (options & SingleCoreApplication::Mode::User) {
server->setSocketOptions(QLocalServer::UserAccessOption);
}
else {
server->setSocketOptions(QLocalServer::WorldAccessOption);
}
server->listen(blockServerName);
QObject::connect(server, &QLocalServer::newConnection, this, &SingleCoreApplicationPrivate::slotConnectionEstablished);
// Reset the number of connections
InstancesInfo* inst = static_cast <InstancesInfo*>(memory->data());
inst->primary = true;
inst->primaryPid = q->applicationPid();
inst->checksum = blockChecksum();
instanceNumber = 0;
}
void SingleCoreApplicationPrivate::startSecondary() {}
void SingleCoreApplicationPrivate::connectToPrimary(const int msecs, const ConnectionType connectionType) {
// Connect to the Local Server of the Primary Instance if not already connected.
if (socket == nullptr) {
socket = new QLocalSocket();
}
// If already connected - we are done;
if (socket->state() == QLocalSocket::ConnectedState)
return;
// If not connect
if (socket->state() == QLocalSocket::UnconnectedState ||
socket->state() == QLocalSocket::ClosingState) {
socket->connectToServer(blockServerName);
}
// Wait for being connected
if (socket->state() == QLocalSocket::ConnectingState) {
socket->waitForConnected(msecs);
}
// Initialisation message according to the SingleCoreApplication protocol
if (socket->state() == QLocalSocket::ConnectedState) {
// Notify the parent that a new instance had been started;
QByteArray initMsg;
QDataStream writeStream(&initMsg, QIODevice::WriteOnly);
writeStream.setVersion(QDataStream::Qt_5_6);
writeStream << blockServerName.toLatin1();
writeStream << static_cast<quint8>(connectionType);
writeStream << instanceNumber;
quint16 checksum = qChecksum(initMsg.constData(), static_cast<quint32>(initMsg.length()));
writeStream << checksum;
// The header indicates the message length that follows
QByteArray header;
QDataStream headerStream(&header, QIODevice::WriteOnly);
headerStream.setVersion(QDataStream::Qt_5_6);
headerStream << static_cast <quint64>(initMsg.length());
socket->write(header);
socket->write(initMsg);
socket->flush();
socket->waitForBytesWritten(msecs);
}
}
quint16 SingleCoreApplicationPrivate::blockChecksum() {
return qChecksum(static_cast <const char*> (memory->data()), offsetof(InstancesInfo, checksum));
}
qint64 SingleCoreApplicationPrivate::primaryPid() {
qint64 pid;
memory->lock();
InstancesInfo* inst = static_cast<InstancesInfo*>(memory->data());
pid = inst->primaryPid;
memory->unlock();
return pid;
return pid;
} }
/** /**
* @brief Executed when a connection has been made to the LocalServer * @brief Executed when a connection has been made to the LocalServer
*/ */
void SingleCoreApplicationPrivate::slotConnectionEstablished() void SingleCoreApplicationPrivate::slotConnectionEstablished() {
{
QLocalSocket *nextConnSocket = server->nextPendingConnection();
connectionMap.insert(nextConnSocket, ConnectionInfo());
QObject::connect(nextConnSocket, &QLocalSocket::aboutToClose, QLocalSocket *nextConnSocket = server->nextPendingConnection();
[nextConnSocket, this]() { connectionMap.insert(nextConnSocket, ConnectionInfo());
auto &info = connectionMap[nextConnSocket];
Q_EMIT this->slotClientConnectionClosed( nextConnSocket, info.instanceId );
}
);
QObject::connect(nextConnSocket, &QLocalSocket::disconnected, QObject::connect(nextConnSocket, &QLocalSocket::aboutToClose,
[nextConnSocket, this](){ [nextConnSocket, this]() {
connectionMap.remove(nextConnSocket); auto &info = connectionMap[nextConnSocket];
nextConnSocket->deleteLater(); Q_EMIT this->slotClientConnectionClosed(nextConnSocket, info.instanceId);
} }
); );
QObject::connect(nextConnSocket, &QLocalSocket::disconnected,
[nextConnSocket, this](){
connectionMap.remove(nextConnSocket);
nextConnSocket->deleteLater();
}
);
QObject::connect(nextConnSocket, &QLocalSocket::readyRead,
[nextConnSocket, this]() {
auto &info = connectionMap[nextConnSocket];
switch(info.stage) {
case StageHeader:
readInitMessageHeader(nextConnSocket);
break;
case StageBody:
readInitMessageBody(nextConnSocket);
break;
case StageConnected:
Q_EMIT this->slotDataAvailable(nextConnSocket, info.instanceId);
break;
default:
break;
};
}
);
QObject::connect(nextConnSocket, &QLocalSocket::readyRead,
[nextConnSocket, this]() {
auto &info = connectionMap[nextConnSocket];
switch(info.stage) {
case StageHeader:
readInitMessageHeader(nextConnSocket);
break;
case StageBody:
readInitMessageBody(nextConnSocket);
break;
case StageConnected:
Q_EMIT this->slotDataAvailable( nextConnSocket, info.instanceId );
break;
default:
break;
};
}
);
} }
void SingleCoreApplicationPrivate::readInitMessageHeader( QLocalSocket *sock ) void SingleCoreApplicationPrivate::readInitMessageHeader(QLocalSocket *sock) {
{
if (!connectionMap.contains( sock )) {
return;
}
if( sock->bytesAvailable() < ( qint64 )sizeof( quint64 ) ) { if (!connectionMap.contains(sock)) {
return; return;
} }
QDataStream headerStream( sock ); if (sock->bytesAvailable() < static_cast<qint64>(sizeof(quint64))) {
return;
}
#if (QT_VERSION >= QT_VERSION_CHECK(5, 6, 0)) QDataStream headerStream(sock);
headerStream.setVersion( QDataStream::Qt_5_6 );
#endif
// Read the header to know the message length headerStream.setVersion(QDataStream::Qt_5_6);
quint64 msgLen = 0;
headerStream >> msgLen; // Read the header to know the message length
ConnectionInfo &info = connectionMap[sock]; quint64 msgLen = 0;
info.stage = StageBody; headerStream >> msgLen;
info.msgLen = msgLen; ConnectionInfo &info = connectionMap[sock];
info.stage = StageBody;
info.msgLen = msgLen;
if (sock->bytesAvailable() >= static_cast<qint64>(msgLen)) {
readInitMessageBody(sock);
}
if ( sock->bytesAvailable() >= (qint64) msgLen ) {
readInitMessageBody( sock );
}
} }
void SingleCoreApplicationPrivate::readInitMessageBody( QLocalSocket *sock ) void SingleCoreApplicationPrivate::readInitMessageBody(QLocalSocket *sock) {
{
Q_Q(SingleCoreApplication);
if (!connectionMap.contains( sock )) { Q_Q(SingleCoreApplication);
return;
}
ConnectionInfo &info = connectionMap[sock]; if (!connectionMap.contains(sock)) {
if( sock->bytesAvailable() < ( qint64 )info.msgLen ) { return;
return; }
}
// Read the message body ConnectionInfo &info = connectionMap[sock];
QByteArray msgBytes = sock->read(info.msgLen); if (sock->bytesAvailable() < static_cast<qint64>(info.msgLen)) {
QDataStream readStream(msgBytes); return;
}
#if (QT_VERSION >= QT_VERSION_CHECK(5, 6, 0)) // Read the message body
readStream.setVersion( QDataStream::Qt_5_6 ); QByteArray msgBytes = sock->read(info.msgLen);
#endif QDataStream readStream(msgBytes);
// server name readStream.setVersion(QDataStream::Qt_5_6);
QByteArray latin1Name;
readStream >> latin1Name;
// connection type // server name
ConnectionType connectionType = InvalidConnection; QByteArray latin1Name;
quint8 connTypeVal = InvalidConnection; readStream >> latin1Name;
readStream >> connTypeVal;
connectionType = static_cast <ConnectionType>( connTypeVal );
// instance id // connection type
quint32 instanceId = 0; ConnectionType connectionType = InvalidConnection;
readStream >> instanceId; quint8 connTypeVal = InvalidConnection;
readStream >> connTypeVal;
connectionType = static_cast <ConnectionType>(connTypeVal);
// checksum // instance id
quint16 msgChecksum = 0; quint32 instanceId = 0;
readStream >> msgChecksum; readStream >> instanceId;
const quint16 actualChecksum = qChecksum( msgBytes.constData(), static_cast<quint32>( msgBytes.length() - sizeof( quint16 ) ) ); // checksum
quint16 msgChecksum = 0;
readStream >> msgChecksum;
bool isValid = readStream.status() == QDataStream::Ok && const quint16 actualChecksum = qChecksum(msgBytes.constData(), static_cast<quint32>(msgBytes.length() - sizeof(quint16)));
QLatin1String(latin1Name) == blockServerName &&
msgChecksum == actualChecksum;
if( !isValid ) { bool isValid = readStream.status() == QDataStream::Ok && QLatin1String(latin1Name) == blockServerName && msgChecksum == actualChecksum;
sock->close();
return;
}
info.instanceId = instanceId; if (!isValid) {
info.stage = StageConnected; sock->close();
return;
}
if( connectionType == NewInstance || info.instanceId = instanceId;
( connectionType == SecondaryInstance && info.stage = StageConnected;
options & SingleCoreApplication::Mode::SecondaryNotification ) )
{ if (connectionType == NewInstance || (connectionType == SecondaryInstance && options & SingleCoreApplication::Mode::SecondaryNotification)) {
Q_EMIT q->instanceStarted(); Q_EMIT q->instanceStarted();
} }
if (sock->bytesAvailable() > 0) {
Q_EMIT this->slotDataAvailable(sock, instanceId);
}
if (sock->bytesAvailable() > 0) {
Q_EMIT this->slotDataAvailable( sock, instanceId );
}
} }
void SingleCoreApplicationPrivate::slotDataAvailable( QLocalSocket *dataSocket, quint32 instanceId ) void SingleCoreApplicationPrivate::slotDataAvailable(QLocalSocket *dataSocket, const quint32 instanceId) {
{
Q_Q(SingleCoreApplication); Q_Q(SingleCoreApplication);
Q_EMIT q->receivedMessage( instanceId, dataSocket->readAll() ); Q_EMIT q->receivedMessage(instanceId, dataSocket->readAll());
} }
void SingleCoreApplicationPrivate::slotClientConnectionClosed( QLocalSocket *closedSocket, quint32 instanceId ) void SingleCoreApplicationPrivate::slotClientConnectionClosed(QLocalSocket *closedSocket, const quint32 instanceId) {
{
if( closedSocket->bytesAvailable() > 0 ) if (closedSocket->bytesAvailable() > 0)
Q_EMIT slotDataAvailable( closedSocket, instanceId ); Q_EMIT slotDataAvailable(closedSocket, instanceId);
} }

View File

@@ -24,76 +24,82 @@
// W A R N I N G !!! // W A R N I N G !!!
// ----------------- // -----------------
// //
// This file is not part of the SingleCoreApplication API. It is used purely as an // This is a modified version of SingleApplication,
// implementation detail. This header file may change from version to // The original version is at:
// version without notice, or may even be removed. //
// https://github.com/itay-grudev/SingleApplication
//
// //
#ifndef SINGLECOREAPPLICATION_P_H #ifndef SINGLECOREAPPLICATION_P_H
#define SINGLECOREAPPLICATION_P_H #define SINGLECOREAPPLICATION_P_H
#include <QtCore/QSharedMemory> #include <QtGlobal>
#include <QtNetwork/QLocalServer> #include <QObject>
#include <QtNetwork/QLocalSocket> #include <QMap>
#include "singlecoreapplication.h" #include "singlecoreapplication.h"
class QLocalServer;
class QLocalSocket;
class QSharedMemory;
struct InstancesInfo { struct InstancesInfo {
bool primary; bool primary;
quint32 secondary; quint32 secondary;
qint64 primaryPid; qint64 primaryPid;
quint16 checksum; quint16 checksum;
}; };
struct ConnectionInfo { struct ConnectionInfo {
explicit ConnectionInfo() : explicit ConnectionInfo() : msgLen(0), instanceId(0), stage(0) {}
msgLen(0), instanceId(0), stage(0) {} qint64 msgLen;
qint64 msgLen; quint32 instanceId;
quint32 instanceId; quint8 stage;
quint8 stage;
}; };
class SingleCoreApplicationPrivate : public QObject { class SingleCoreApplicationPrivate : public QObject {
Q_OBJECT Q_OBJECT
public: public:
enum ConnectionType : quint8 { enum ConnectionType : quint8 {
InvalidConnection = 0, InvalidConnection = 0,
NewInstance = 1, NewInstance = 1,
SecondaryInstance = 2, SecondaryInstance = 2,
Reconnect = 3 Reconnect = 3
}; };
enum ConnectionStage : quint8 { enum ConnectionStage : quint8 {
StageHeader = 0, StageHeader = 0,
StageBody = 1, StageBody = 1,
StageConnected = 2, StageConnected = 2,
}; };
Q_DECLARE_PUBLIC(SingleCoreApplication) Q_DECLARE_PUBLIC(SingleCoreApplication)
SingleCoreApplicationPrivate( SingleCoreApplication *q_ptr ); explicit SingleCoreApplicationPrivate(SingleCoreApplication *_q_ptr);
~SingleCoreApplicationPrivate(); ~SingleCoreApplicationPrivate() override;
void genBlockServerName(); void genBlockServerName();
void initializeMemoryBlock(); void initializeMemoryBlock();
void startPrimary(); void startPrimary();
void startSecondary(); void startSecondary();
void connectToPrimary(int msecs, ConnectionType connectionType ); void connectToPrimary(const int msecs, const ConnectionType connectionType);
quint16 blockChecksum(); quint16 blockChecksum();
qint64 primaryPid(); qint64 primaryPid();
void readInitMessageHeader(QLocalSocket *socket); void readInitMessageHeader(QLocalSocket *socket);
void readInitMessageBody(QLocalSocket *socket); void readInitMessageBody(QLocalSocket *socket);
SingleCoreApplication *q_ptr; SingleCoreApplication *q_ptr;
QSharedMemory *memory; QSharedMemory *memory;
QLocalSocket *socket; QLocalSocket *socket;
QLocalServer *server; QLocalServer *server;
quint32 instanceNumber; quint32 instanceNumber;
QString blockServerName; QString blockServerName;
SingleCoreApplication::Options options; SingleCoreApplication::Options options;
QMap<QLocalSocket*, ConnectionInfo> connectionMap; QMap<QLocalSocket*, ConnectionInfo> connectionMap;
public Q_SLOTS: public slots:
void slotConnectionEstablished(); void slotConnectionEstablished();
void slotDataAvailable( QLocalSocket*, quint32 ); void slotDataAvailable(QLocalSocket*, const quint32);
void slotClientConnectionClosed( QLocalSocket*, quint32 ); void slotClientConnectionClosed(QLocalSocket*, const quint32);
}; };
#endif // SINGLECOREAPPLICATION_P_H #endif // SINGLECOREAPPLICATION_P_H

View File

@@ -1,8 +1,4 @@
cmake_minimum_required(VERSION 2.8.11) cmake_minimum_required(VERSION 3.0)
set(CMAKE_C_FLAGS "${CMAKE_C_FLAGS} -Wall")
set(CMAKE_CXX_STANDARD 11)
set(CMAKE_CXX_FLAGS "${CMAKE_CXX_FLAGS} --std=c++11 -U__STRICT_ANSI__ -fpermissive -Wall -Woverloaded-virtual -Wno-sign-compare -Wno-deprecated-declarations -Wno-unused-local-typedefs -Wno-delete-non-virtual-dtor")
set(TAGLIB_SOVERSION_CURRENT 17) set(TAGLIB_SOVERSION_CURRENT 17)
set(TAGLIB_SOVERSION_REVISION 0) set(TAGLIB_SOVERSION_REVISION 0)
@@ -12,70 +8,24 @@ math(EXPR TAGLIB_SOVERSION_MAJOR "${TAGLIB_SOVERSION_CURRENT} - ${TAGLIB_SOVERSI
math(EXPR TAGLIB_SOVERSION_MINOR "${TAGLIB_SOVERSION_AGE}") math(EXPR TAGLIB_SOVERSION_MINOR "${TAGLIB_SOVERSION_AGE}")
math(EXPR TAGLIB_SOVERSION_PATCH "${TAGLIB_SOVERSION_REVISION}") math(EXPR TAGLIB_SOVERSION_PATCH "${TAGLIB_SOVERSION_REVISION}")
include(TestBigEndian)
test_big_endian(IS_BIG_ENDIAN)
if(NOT IS_BIG_ENDIAN)
add_definitions(-DSYSTEM_BYTEORDER=1)
else()
add_definitions(-DSYSTEM_BYTEORDER=2)
endif()
include(ConfigureChecks.cmake) include(ConfigureChecks.cmake)
configure_file(config.h.cmake "${CMAKE_CURRENT_BINARY_DIR}/config.h") set(TESTS_DIR "${CMAKE_SOURCE_DIR}/tests/taglib/")
configure_file(taglib_config.h.cmake ${CMAKE_CURRENT_BINARY_DIR}/taglib_config.h) configure_file(taglib-config.h.cmake "${CMAKE_CURRENT_BINARY_DIR}/taglib-config.h")
add_definitions(-DHAVE_CONFIG_H) add_definitions(-DHAVE_CONFIG_H)
add_definitions(-DTAGLIB_STATIC) add_definitions(-DTAGLIB_STATIC)
set(CMAKE_INCLUDE_CURRENT_DIR ON)
include_directories(
${CMAKE_CURRENT_SOURCE_DIR}/toolkit
${CMAKE_CURRENT_SOURCE_DIR}/asf
${CMAKE_CURRENT_SOURCE_DIR}/mpeg
${CMAKE_CURRENT_SOURCE_DIR}/ogg
${CMAKE_CURRENT_SOURCE_DIR}/ogg/flac
${CMAKE_CURRENT_SOURCE_DIR}/flac
${CMAKE_CURRENT_SOURCE_DIR}/mpc
${CMAKE_CURRENT_SOURCE_DIR}/mp4
${CMAKE_CURRENT_SOURCE_DIR}/ogg/vorbis
${CMAKE_CURRENT_SOURCE_DIR}/ogg/speex
${CMAKE_CURRENT_SOURCE_DIR}/ogg/opus
${CMAKE_CURRENT_SOURCE_DIR}/mpeg/id3v2
${CMAKE_CURRENT_SOURCE_DIR}/mpeg/id3v2/frames
${CMAKE_CURRENT_SOURCE_DIR}/mpeg/id3v1
${CMAKE_CURRENT_SOURCE_DIR}/ape
${CMAKE_CURRENT_SOURCE_DIR}/wavpack
${CMAKE_CURRENT_SOURCE_DIR}/trueaudio
${CMAKE_CURRENT_SOURCE_DIR}/riff
${CMAKE_CURRENT_SOURCE_DIR}/riff/aiff
${CMAKE_CURRENT_SOURCE_DIR}/riff/wav
${CMAKE_CURRENT_SOURCE_DIR}/mod
${CMAKE_CURRENT_SOURCE_DIR}/s3m
${CMAKE_CURRENT_SOURCE_DIR}/it
${CMAKE_CURRENT_SOURCE_DIR}/xm
${CMAKE_CURRENT_SOURCE_DIR}/dsf
${CMAKE_CURRENT_SOURCE_DIR}/dsdiff
${CMAKE_SOURCE_DIR}/3rdparty
)
if(ZLIB_FOUND)
include_directories(${ZLIB_INCLUDE_DIR})
elseif(HAVE_ZLIB_SOURCE)
include_directories(${ZLIB_SOURCE})
endif()
set(tag_HDRS set(tag_HDRS
tag.h tag.h
fileref.h fileref.h
audioproperties.h audioproperties.h
taglib_export.h taglib_export.h
${CMAKE_CURRENT_BINARY_DIR}/taglib_config.h
toolkit/taglib.h toolkit/taglib.h
toolkit/tstring.h toolkit/tstring.h
toolkit/tlist.h toolkit/tlist.h
toolkit/tlist.tcc toolkit/tlist.tcc
toolkit/tstringlist.h toolkit/tstringlist.h
toolkit/tstringhandler.h
toolkit/tbytevector.h toolkit/tbytevector.h
toolkit/tbytevectorlist.h toolkit/tbytevectorlist.h
toolkit/tbytevectorstream.h toolkit/tbytevectorstream.h
@@ -84,6 +34,8 @@ set(tag_HDRS
toolkit/tfilestream.h toolkit/tfilestream.h
toolkit/tmap.h toolkit/tmap.h
toolkit/tmap.tcc toolkit/tmap.tcc
toolkit/tpicture.h
toolkit/tpicturemap.h
toolkit/tpropertymap.h toolkit/tpropertymap.h
toolkit/trefcounter.h toolkit/trefcounter.h
toolkit/tdebuglistener.h toolkit/tdebuglistener.h
@@ -93,6 +45,7 @@ set(tag_HDRS
mpeg/xingheader.h mpeg/xingheader.h
mpeg/id3v1/id3v1tag.h mpeg/id3v1/id3v1tag.h
mpeg/id3v1/id3v1genres.h mpeg/id3v1/id3v1genres.h
mpeg/id3v2/id3v2.h
mpeg/id3v2/id3v2extendedheader.h mpeg/id3v2/id3v2extendedheader.h
mpeg/id3v2/id3v2frame.h mpeg/id3v2/id3v2frame.h
mpeg/id3v2/id3v2header.h mpeg/id3v2/id3v2header.h
@@ -342,8 +295,10 @@ set(dsdiff_SRCS
) )
set(toolkit_SRCS set(toolkit_SRCS
toolkit/taglib.cpp
toolkit/tstring.cpp toolkit/tstring.cpp
toolkit/tstringlist.cpp toolkit/tstringlist.cpp
toolkit/tstringhandler.cpp
toolkit/tbytevector.cpp toolkit/tbytevector.cpp
toolkit/tbytevectorlist.cpp toolkit/tbytevectorlist.cpp
toolkit/tbytevectorstream.cpp toolkit/tbytevectorstream.cpp
@@ -351,28 +306,40 @@ set(toolkit_SRCS
toolkit/tfile.cpp toolkit/tfile.cpp
toolkit/tfilestream.cpp toolkit/tfilestream.cpp
toolkit/tdebug.cpp toolkit/tdebug.cpp
toolkit/tpicture.cpp
toolkit/tpicturemap.cpp
toolkit/tpropertymap.cpp toolkit/tpropertymap.cpp
toolkit/trefcounter.cpp toolkit/trefcounter.cpp
toolkit/tdebuglistener.cpp toolkit/tdebuglistener.cpp
toolkit/tzlib.cpp toolkit/tzlib.cpp
) )
if(HAVE_ZLIB_SOURCE)
set(zlib_SRCS
${ZLIB_SOURCE}/adler32.c
${ZLIB_SOURCE}/crc32.c
${ZLIB_SOURCE}/inffast.c
${ZLIB_SOURCE}/inflate.c
${ZLIB_SOURCE}/inftrees.c
${ZLIB_SOURCE}/zutil.c
)
endif()
set(tag_LIB_SRCS set(tag_LIB_SRCS
${mpeg_SRCS} ${id3v1_SRCS} ${id3v2_SRCS} ${frames_SRCS} ${ogg_SRCS} ${mpeg_SRCS}
${vorbis_SRCS} ${oggflacs_SRCS} ${mpc_SRCS} ${ape_SRCS} ${toolkit_SRCS} ${flacs_SRCS} ${id3v1_SRCS}
${wavpack_SRCS} ${speex_SRCS} ${trueaudio_SRCS} ${riff_SRCS} ${aiff_SRCS} ${wav_SRCS} ${id3v2_SRCS}
${asf_SRCS} ${mp4_SRCS} ${mod_SRCS} ${s3m_SRCS} ${it_SRCS} ${xm_SRCS} ${opus_SRCS} ${dsf_SRCS} ${dsdiff_SRCS} ${frames_SRCS}
${ogg_SRCS}
${vorbis_SRCS}
${oggflacs_SRCS}
${mpc_SRCS}
${ape_SRCS}
${toolkit_SRCS}
${flacs_SRCS}
${wavpack_SRCS}
${speex_SRCS}
${trueaudio_SRCS}
${riff_SRCS}
${aiff_SRCS} ${wav_SRCS}
${asf_SRCS}
${mp4_SRCS}
${mod_SRCS}
${s3m_SRCS}
${it_SRCS}
${xm_SRCS}
${opus_SRCS}
${dsf_SRCS}
${dsdiff_SRCS}
${zlib_SRCS} ${zlib_SRCS}
tag.cpp tag.cpp
tagunion.cpp tagunion.cpp
@@ -383,9 +350,40 @@ set(tag_LIB_SRCS
add_library(tag STATIC ${tag_LIB_SRCS} ${tag_HDRS}) add_library(tag STATIC ${tag_LIB_SRCS} ${tag_HDRS})
if(HAVE_ZLIB AND NOT HAVE_ZLIB_SOURCE) target_include_directories(tag PRIVATE
target_link_libraries(tag ${ZLIB_LIBRARIES}) ${ZLIB_INCLUDE_DIR}
endif() ${CMAKE_CURRENT_BINARY_DIR}
${CMAKE_CURRENT_SOURCE_DIR}
${CMAKE_CURRENT_SOURCE_DIR}/toolkit
${CMAKE_CURRENT_SOURCE_DIR}/asf
${CMAKE_CURRENT_SOURCE_DIR}/mpeg
${CMAKE_CURRENT_SOURCE_DIR}/ogg
${CMAKE_CURRENT_SOURCE_DIR}/ogg/flac
${CMAKE_CURRENT_SOURCE_DIR}/flac
${CMAKE_CURRENT_SOURCE_DIR}/mpc
${CMAKE_CURRENT_SOURCE_DIR}/mp4
${CMAKE_CURRENT_SOURCE_DIR}/ogg/vorbis
${CMAKE_CURRENT_SOURCE_DIR}/ogg/speex
${CMAKE_CURRENT_SOURCE_DIR}/ogg/opus
${CMAKE_CURRENT_SOURCE_DIR}/mpeg/id3v2
${CMAKE_CURRENT_SOURCE_DIR}/mpeg/id3v2/frames
${CMAKE_CURRENT_SOURCE_DIR}/mpeg/id3v1
${CMAKE_CURRENT_SOURCE_DIR}/ape
${CMAKE_CURRENT_SOURCE_DIR}/wavpack
${CMAKE_CURRENT_SOURCE_DIR}/trueaudio
${CMAKE_CURRENT_SOURCE_DIR}/riff
${CMAKE_CURRENT_SOURCE_DIR}/riff/aiff
${CMAKE_CURRENT_SOURCE_DIR}/riff/wav
${CMAKE_CURRENT_SOURCE_DIR}/mod
${CMAKE_CURRENT_SOURCE_DIR}/s3m
${CMAKE_CURRENT_SOURCE_DIR}/it
${CMAKE_CURRENT_SOURCE_DIR}/xm
${CMAKE_CURRENT_SOURCE_DIR}/dsf
${CMAKE_CURRENT_SOURCE_DIR}/dsdiff
${CMAKE_SOURCE_DIR}/3rdparty
)
target_link_libraries(tag PRIVATE ${ZLIB_LIBRARIES})
set_target_properties(tag PROPERTIES set_target_properties(tag PROPERTIES
VERSION ${TAGLIB_SOVERSION_MAJOR}.${TAGLIB_SOVERSION_MINOR}.${TAGLIB_SOVERSION_PATCH} VERSION ${TAGLIB_SOVERSION_MAJOR}.${TAGLIB_SOVERSION_MINOR}.${TAGLIB_SOVERSION_PATCH}
@@ -402,4 +400,3 @@ foreach(header ${tag_HDRS})
COPYONLY COPYONLY
) )
endforeach() endforeach()

View File

@@ -1,36 +1,37 @@
include(CheckLibraryExists) include(CheckLibraryExists)
include(CheckTypeSize) include(CheckTypeSize)
include(CheckCXXCompilerFlag)
include(CheckCXXSourceCompiles) include(CheckCXXSourceCompiles)
# Check if the size of numeric types are suitable. # Check if the size of numeric types are suitable.
check_type_size("short" SIZEOF_SHORT) check_type_size("short" SIZEOF_SHORT)
if(NOT ${SIZEOF_SHORT} EQUAL 2) if(NOT SIZEOF_SHORT EQUAL 2)
message(FATAL_ERROR "TagLib requires that short is 16-bit wide.") message(FATAL_ERROR "TagLib requires that short is 16-bit wide.")
endif() endif()
check_type_size("int" SIZEOF_INT) check_type_size("int" SIZEOF_INT)
if(NOT ${SIZEOF_INT} EQUAL 4) if(NOT SIZEOF_INT EQUAL 4)
message(FATAL_ERROR "TagLib requires that int is 32-bit wide.") message(FATAL_ERROR "TagLib requires that int is 32-bit wide.")
endif() endif()
check_type_size("long long" SIZEOF_LONGLONG) check_type_size("long long" SIZEOF_LONGLONG)
if(NOT ${SIZEOF_LONGLONG} EQUAL 8) if(NOT SIZEOF_LONGLONG EQUAL 8)
message(FATAL_ERROR "TagLib requires that long long is 64-bit wide.") message(FATAL_ERROR "TagLib requires that long long is 64-bit wide.")
endif() endif()
check_type_size("wchar_t" SIZEOF_WCHAR_T) check_type_size("wchar_t" SIZEOF_WCHAR_T)
if(${SIZEOF_WCHAR_T} LESS 2) if(SIZEOF_WCHAR_T LESS 2)
message(FATAL_ERROR "TagLib requires that wchar_t is sufficient to store a UTF-16 char.") message(FATAL_ERROR "TagLib requires that wchar_t is sufficient to store a UTF-16 char.")
endif() endif()
check_type_size("float" SIZEOF_FLOAT) check_type_size("float" SIZEOF_FLOAT)
if(NOT ${SIZEOF_FLOAT} EQUAL 4) if(NOT SIZEOF_FLOAT EQUAL 4)
message(FATAL_ERROR "TagLib requires that float is 32-bit wide.") message(FATAL_ERROR "TagLib requires that float is 32-bit wide.")
endif() endif()
check_type_size("double" SIZEOF_DOUBLE) check_type_size("double" SIZEOF_DOUBLE)
if(NOT ${SIZEOF_DOUBLE} EQUAL 8) if(NOT SIZEOF_DOUBLE EQUAL 8)
message(FATAL_ERROR "TagLib requires that double is 64-bit wide.") message(FATAL_ERROR "TagLib requires that double is 64-bit wide.")
endif() endif()
@@ -211,5 +212,5 @@ endif()
# Detect WinRT mode # Detect WinRT mode
if(CMAKE_SYSTEM_NAME STREQUAL "WindowsStore") if(CMAKE_SYSTEM_NAME STREQUAL "WindowsStore")
set(PLATFORM WINRT 1) set(PLATFORM WINRT 1)
endif() endif()

View File

@@ -31,14 +31,16 @@
* http://www.mozilla.org/MPL/ * * http://www.mozilla.org/MPL/ *
***************************************************************************/ ***************************************************************************/
#include <tbytevector.h> #include <memory>
#include <tstring.h>
#include <tdebug.h> #include "tbytevector.h"
#include <tagunion.h> #include "tstring.h"
#include <id3v1tag.h> #include "tdebug.h"
#include <id3v2header.h> #include "tagunion.h"
#include <tpropertymap.h> #include "id3v1tag.h"
#include <tagutils.h> #include "id3v2header.h"
#include "tpropertymap.h"
#include "tagutils.h"
#include "apefile.h" #include "apefile.h"
#include "apetag.h" #include "apetag.h"
@@ -46,122 +48,98 @@
using namespace Strawberry_TagLib::TagLib; using namespace Strawberry_TagLib::TagLib;
namespace namespace {
{ enum { ApeAPEIndex = 0,
enum { ApeAPEIndex = 0, ApeID3v1Index = 1 }; ApeID3v1Index = 1 };
} }
class APE::File::FilePrivate class APE::File::FilePrivate {
{ public:
public: FilePrivate() : APELocation(-1),
FilePrivate() : APESize(0),
APELocation(-1), ID3v1Location(-1),
APESize(0), ID3v2Location(-1),
ID3v1Location(-1), ID3v2Size(0) {}
ID3v2Header(0),
ID3v2Location(-1),
ID3v2Size(0),
properties(0) {}
~FilePrivate() long long APELocation;
{ long long APESize;
delete ID3v2Header;
delete properties;
}
long APELocation; long long ID3v1Location;
long APESize;
long ID3v1Location; std::unique_ptr<ID3v2::Header> ID3v2Header;
long long ID3v2Location;
long long ID3v2Size;
ID3v2::Header *ID3v2Header; DoubleTagUnion tag;
long ID3v2Location;
long ID3v2Size;
TagUnion tag; std::unique_ptr<AudioProperties> properties;
Properties *properties;
}; };
//////////////////////////////////////////////////////////////////////////////// ////////////////////////////////////////////////////////////////////////////////
// static members // static members
//////////////////////////////////////////////////////////////////////////////// ////////////////////////////////////////////////////////////////////////////////
bool APE::File::isSupported(IOStream *stream) bool APE::File::isSupported(IOStream *stream) {
{
// An APE file has an ID "MAC " somewhere. An ID3v2 tag may precede. // An APE file has an ID "MAC " somewhere. An ID3v2 tag may precede.
const ByteVector buffer = Utils::readHeader(stream, bufferSize(), true); const ByteVector buffer = Utils::readHeader(stream, bufferSize(), true);
return (buffer.find("MAC ") >= 0); return (buffer.find("MAC ") == 0);
} }
//////////////////////////////////////////////////////////////////////////////// ////////////////////////////////////////////////////////////////////////////////
// public members // public members
//////////////////////////////////////////////////////////////////////////////// ////////////////////////////////////////////////////////////////////////////////
APE::File::File(FileName file, bool readProperties, Properties::ReadStyle) : APE::File::File(FileName file, bool readProperties, AudioProperties::ReadStyle) : Strawberry_TagLib::TagLib::File(file), d(new FilePrivate()) {
Strawberry_TagLib::TagLib::File(file),
d(new FilePrivate()) if (isOpen())
{
if(isOpen())
read(readProperties); read(readProperties);
} }
APE::File::File(IOStream *stream, bool readProperties, Properties::ReadStyle) : APE::File::File(IOStream *stream, bool readProperties, AudioProperties::ReadStyle) : Strawberry_TagLib::TagLib::File(stream), d(new FilePrivate()) {
Strawberry_TagLib::TagLib::File(stream),
d(new FilePrivate()) if (isOpen())
{
if(isOpen())
read(readProperties); read(readProperties);
} }
APE::File::~File() APE::File::~File() {
{
delete d; delete d;
} }
Strawberry_TagLib::TagLib::Tag *APE::File::tag() const Strawberry_TagLib::TagLib::Tag *APE::File::tag() const {
{
return &d->tag; return &d->tag;
} }
PropertyMap APE::File::properties() const PropertyMap APE::File::setProperties(const PropertyMap &properties) {
{
return d->tag.properties();
}
void APE::File::removeUnsupportedProperties(const StringList &properties) if (ID3v1Tag())
{
d->tag.removeUnsupportedProperties(properties);
}
PropertyMap APE::File::setProperties(const PropertyMap &properties)
{
if(ID3v1Tag())
ID3v1Tag()->setProperties(properties); ID3v1Tag()->setProperties(properties);
return APETag(true)->setProperties(properties); return APETag(true)->setProperties(properties);
} }
APE::Properties *APE::File::audioProperties() const APE::AudioProperties *APE::File::audioProperties() const {
{ return d->properties.get();
return d->properties;
} }
bool APE::File::save() bool APE::File::save() {
{
if(readOnly()) { if (readOnly()) {
debug("APE::File::save() -- File is read only."); debug("APE::File::save() -- File is read only.");
return false; return false;
} }
// Update ID3v1 tag // Update ID3v1 tag
if(ID3v1Tag() && !ID3v1Tag()->isEmpty()) { if (ID3v1Tag() && !ID3v1Tag()->isEmpty()) {
// ID3v1 tag is not empty. Update the old one or create a new one. // ID3v1 tag is not empty. Update the old one or create a new one.
if(d->ID3v1Location >= 0) { if (d->ID3v1Location >= 0) {
seek(d->ID3v1Location); seek(d->ID3v1Location);
} }
else { else {
@@ -175,7 +153,7 @@ bool APE::File::save()
// ID3v1 tag is empty. Remove the old one. // ID3v1 tag is empty. Remove the old one.
if(d->ID3v1Location >= 0) { if (d->ID3v1Location >= 0) {
truncate(d->ID3v1Location); truncate(d->ID3v1Location);
d->ID3v1Location = -1; d->ID3v1Location = -1;
} }
@@ -183,12 +161,12 @@ bool APE::File::save()
// Update APE tag // Update APE tag
if(APETag() && !APETag()->isEmpty()) { if (APETag() && !APETag()->isEmpty()) {
// APE tag is not empty. Update the old one or create a new one. // APE tag is not empty. Update the old one or create a new one.
if(d->APELocation < 0) { if (d->APELocation < 0) {
if(d->ID3v1Location >= 0) if (d->ID3v1Location >= 0)
d->APELocation = d->ID3v1Location; d->APELocation = d->ID3v1Location;
else else
d->APELocation = length(); d->APELocation = length();
@@ -197,7 +175,7 @@ bool APE::File::save()
const ByteVector data = APETag()->render(); const ByteVector data = APETag()->render();
insert(data, d->APELocation, d->APESize); insert(data, d->APELocation, d->APESize);
if(d->ID3v1Location >= 0) if (d->ID3v1Location >= 0)
d->ID3v1Location += (static_cast<long>(data.size()) - d->APESize); d->ID3v1Location += (static_cast<long>(data.size()) - d->APESize);
d->APESize = data.size(); d->APESize = data.size();
@@ -206,10 +184,10 @@ bool APE::File::save()
// APE tag is empty. Remove the old one. // APE tag is empty. Remove the old one.
if(d->APELocation >= 0) { if (d->APELocation >= 0) {
removeBlock(d->APELocation, d->APESize); removeBlock(d->APELocation, d->APESize);
if(d->ID3v1Location >= 0) if (d->ID3v1Location >= 0)
d->ID3v1Location -= d->APESize; d->ID3v1Location -= d->APESize;
d->APELocation = -1; d->APELocation = -1;
@@ -218,37 +196,35 @@ bool APE::File::save()
} }
return true; return true;
} }
ID3v1::Tag *APE::File::ID3v1Tag(bool create) ID3v1::Tag *APE::File::ID3v1Tag(bool create) {
{
return d->tag.access<ID3v1::Tag>(ApeID3v1Index, create); return d->tag.access<ID3v1::Tag>(ApeID3v1Index, create);
} }
APE::Tag *APE::File::APETag(bool create) APE::Tag *APE::File::APETag(bool create) {
{
return d->tag.access<APE::Tag>(ApeAPEIndex, create); return d->tag.access<APE::Tag>(ApeAPEIndex, create);
} }
void APE::File::strip(int tags) void APE::File::strip(int tags) {
{
if(tags & ID3v1)
d->tag.set(ApeID3v1Index, 0);
if(tags & APE) if (tags & ID3v1)
d->tag.set(ApeAPEIndex, 0); d->tag.set(ApeID3v1Index, nullptr);
if(!ID3v1Tag()) if (tags & APE)
d->tag.set(ApeAPEIndex, nullptr);
if (!ID3v1Tag())
APETag(true); APETag(true);
} }
bool APE::File::hasAPETag() const bool APE::File::hasAPETag() const {
{
return (d->APELocation >= 0); return (d->APELocation >= 0);
} }
bool APE::File::hasID3v1Tag() const bool APE::File::hasID3v1Tag() const {
{
return (d->ID3v1Location >= 0); return (d->ID3v1Location >= 0);
} }
@@ -256,15 +232,15 @@ bool APE::File::hasID3v1Tag() const
// private members // private members
//////////////////////////////////////////////////////////////////////////////// ////////////////////////////////////////////////////////////////////////////////
void APE::File::read(bool readProperties) void APE::File::read(bool readProperties) {
{
// Look for an ID3v2 tag // Look for an ID3v2 tag
d->ID3v2Location = Utils::findID3v2(this); d->ID3v2Location = Utils::findID3v2(this);
if(d->ID3v2Location >= 0) { if (d->ID3v2Location >= 0) {
seek(d->ID3v2Location); seek(d->ID3v2Location);
d->ID3v2Header = new ID3v2::Header(readBlock(ID3v2::Header::size())); d->ID3v2Header.reset(new ID3v2::Header(readBlock(ID3v2::Header::size())));
d->ID3v2Size = d->ID3v2Header->completeTagSize(); d->ID3v2Size = d->ID3v2Header->completeTagSize();
} }
@@ -272,36 +248,36 @@ void APE::File::read(bool readProperties)
d->ID3v1Location = Utils::findID3v1(this); d->ID3v1Location = Utils::findID3v1(this);
if(d->ID3v1Location >= 0) if (d->ID3v1Location >= 0)
d->tag.set(ApeID3v1Index, new ID3v1::Tag(this, d->ID3v1Location)); d->tag.set(ApeID3v1Index, new ID3v1::Tag(this, d->ID3v1Location));
// Look for an APE tag // Look for an APE tag
d->APELocation = Utils::findAPE(this, d->ID3v1Location); d->APELocation = Utils::findAPE(this, d->ID3v1Location);
if(d->APELocation >= 0) { if (d->APELocation >= 0) {
d->tag.set(ApeAPEIndex, new APE::Tag(this, d->APELocation)); d->tag.set(ApeAPEIndex, new APE::Tag(this, d->APELocation));
d->APESize = APETag()->footer()->completeTagSize(); d->APESize = APETag()->footer()->completeTagSize();
d->APELocation = d->APELocation + APE::Footer::size() - d->APESize; d->APELocation = d->APELocation + APE::Footer::size() - d->APESize;
} }
if(d->ID3v1Location < 0) if (d->ID3v1Location < 0)
APETag(true); APETag(true);
// Look for APE audio properties // Look for APE audio properties
if(readProperties) { if (readProperties) {
long streamLength; long long streamLength;
if(d->APELocation >= 0) if (d->APELocation >= 0)
streamLength = d->APELocation; streamLength = d->APELocation;
else if(d->ID3v1Location >= 0) else if (d->ID3v1Location >= 0)
streamLength = d->ID3v1Location; streamLength = d->ID3v1Location;
else else
streamLength = length(); streamLength = length();
if(d->ID3v2Location >= 0) { if (d->ID3v2Location >= 0) {
seek(d->ID3v2Location + d->ID3v2Size); seek(d->ID3v2Location + d->ID3v2Size);
streamLength -= (d->ID3v2Location + d->ID3v2Size); streamLength -= (d->ID3v2Location + d->ID3v2Size);
} }
@@ -309,6 +285,7 @@ void APE::File::read(bool readProperties)
seek(0); seek(0);
} }
d->properties = new Properties(this, streamLength); d->properties.reset(new AudioProperties(this, streamLength));
} }
} }

View File

@@ -41,197 +41,174 @@
namespace Strawberry_TagLib { namespace Strawberry_TagLib {
namespace TagLib { namespace TagLib {
class Tag; class Tag;
namespace ID3v1 { class Tag; } namespace ID3v1 {
namespace APE { class Tag; } class Tag;
}
namespace APE {
class Tag;
}
//! An implementation of APE metadata //! An implementation of APE metadata
/*!
* This is implementation of APE metadata.
*
* This supports ID3v1 and APE (v1 and v2) style comments as well as reading stream properties from the file.
*
*/
namespace APE {
//! An implementation of TagLib::File with APE specific methods
/*!
* This implements and provides an interface for APE files to the
* TagLib::Tag and TagLib::AudioProperties interfaces by way of implementing
* the abstract TagLib::File API as well as providing some additional information specific to APE files.
*
*/
class TAGLIB_EXPORT File : public Strawberry_TagLib::TagLib::File {
public:
/*!
* This set of flags is used for various operations and is suitable for being OR-ed together.
*/
enum TagTypes {
//! Empty set. Matches no tag types.
NoTags = 0x0000,
//! Matches ID3v1 tags.
ID3v1 = 0x0001,
//! Matches APE tags.
APE = 0x0002,
//! Matches all tag types.
AllTags = 0xffff
};
/*! /*!
* This is implementation of APE metadata. * Constructs an APE file from \a file.
* If \a readProperties is true the file's audio properties will also be read.
* *
* This supports ID3v1 and APE (v1 and v2) style comments as well as reading stream * \note In the current implementation, \a propertiesStyle is ignored.
* properties from the file.
*/ */
explicit File(FileName file, bool readProperties = true, AudioProperties::ReadStyle propertiesStyle = AudioProperties::Average);
namespace APE { /*!
* Constructs an APE file from \a stream.
* If \a readProperties is true the file's audio properties will also be read.
*
* \note TagLib will *not* take ownership of the stream, the caller is responsible for deleting it after the File object.
*
* \note In the current implementation, \a propertiesStyle is ignored.
*/
explicit File(IOStream *stream, bool readProperties = true, AudioProperties::ReadStyle propertiesStyle = AudioProperties::Average);
//! An implementation of Strawberry_TagLib::TagLib::File with APE specific methods /*!
* Destroys this instance of the File.
*/
~File() override;
/*! /*!
* This implements and provides an interface for APE files to the * Returns the Tag for this file. This will be an APE tag, an ID3v1 tag or a combination of the two.
* Strawberry_TagLib::TagLib::Tag and Strawberry_TagLib::TagLib::AudioProperties interfaces by way of implementing */
* the abstract Strawberry_TagLib::TagLib::File API as well as providing some additional Strawberry_TagLib::TagLib::Tag *tag() const override;
* information specific to APE files.
*/
class TAGLIB_EXPORT File : public Strawberry_TagLib::TagLib::File /*!
{ * Implements the unified property interface -- import function.
public: * Creates an APEv2 tag if necessary.
/*! * A potentially existing ID3v1 tag will be updated as well.
* This set of flags is used for various operations and is suitable for */
* being OR-ed together. PropertyMap setProperties(const PropertyMap&) override;
*/
enum TagTypes {
//! Empty set. Matches no tag types.
NoTags = 0x0000,
//! Matches ID3v1 tags.
ID3v1 = 0x0001,
//! Matches APE tags.
APE = 0x0002,
//! Matches all tag types.
AllTags = 0xffff
};
/*! /*!
* Constructs an APE file from \a file. If \a readProperties is true the * Returns the APE::AudioProperties for this file.
* file's audio properties will also be read. * If no audio properties were read then this will return a null pointer.
* */
* \note In the current implementation, \a propertiesStyle is ignored. AudioProperties *audioProperties() const override;
*/
File(FileName file, bool readProperties = true,
Properties::ReadStyle propertiesStyle = Properties::Average);
/*! /*!
* Constructs an APE file from \a stream. If \a readProperties is true the * Saves the file.
* file's audio properties will also be read. *
* * \note According to the official Monkey's Audio SDK, an APE file
* \note TagLib will *not* take ownership of the stream, the caller is * can only have either ID3V1 or APE tags, so a parameter is used here.
* responsible for deleting it after the File object. */
* bool save() override;
* \note In the current implementation, \a propertiesStyle is ignored.
*/
File(IOStream *stream, bool readProperties = true,
Properties::ReadStyle propertiesStyle = Properties::Average);
/*! /*!
* Destroys this instance of the File. * Returns a pointer to the ID3v1 tag of the file.
*/ *
virtual ~File(); * If \a create is false (the default) this may return a null pointer if there is no valid ID3v1 tag.
* If \a create is true it will create an ID3v1 tag if one does not exist and returns a valid pointer.
*
* \note This may return a valid pointer regardless of whether or not the file on disk has an ID3v1 tag.
* Use hasID3v1Tag() to check if the file on disk actually has an ID3v1 tag.
*
* \note The Tag <b>is still</b> owned by the MPEG::File and should not be deleted by the user.
* It will be deleted when the file (object) is destroyed.
*
* \see hasID3v1Tag()
*/
ID3v1::Tag *ID3v1Tag(bool create = false);
/*! /*!
* Returns the Tag for this file. This will be an APE tag, an ID3v1 tag * Returns a pointer to the APE tag of the file.
* or a combination of the two. *
*/ * If \a create is false (the default) this may return a null pointer if there is no valid APE tag.
virtual Strawberry_TagLib::TagLib::Tag *tag() const; * If \a create is true it will create an APE tag if one does not exist and returns a valid pointer.
*
* \note This may return a valid pointer regardless of whether or not the file on disk has an APE tag.
* Use hasAPETag() to check if the file on disk actually has an APE tag.
*
* \note The Tag <b>is still</b> owned by the MPEG::File and should not be deleted by the user.
* It will be deleted when the file (object) is destroyed.
*
* \see hasAPETag()
*/
APE::Tag *APETag(bool create = false);
/*! /*!
* Implements the unified property interface -- export function. * This will remove the tags that match the OR-ed together TagTypes from the file.
* If the file contains both an APE and an ID3v1 tag, only APE * By default it removes all tags.
* will be converted to the PropertyMap. *
*/ * \note This will also invalidate pointers to the tags as their memory will be freed.
PropertyMap properties() const; * \note In order to make the removal permanent save() still needs to be called
*/
void strip(int tags = AllTags);
/*! /*!
* Removes unsupported properties. Forwards to the actual Tag's * Returns whether or not the file on disk actually has an APE tag.
* removeUnsupportedProperties() function. *
*/ * \see APETag()
void removeUnsupportedProperties(const StringList &properties); */
bool hasAPETag() const;
/*! /*!
* Implements the unified property interface -- import function. * Returns whether or not the file on disk actually has an ID3v1 tag.
* Creates an APEv2 tag if necessary. A potentially existing ID3v1 *
* tag will be updated as well. * \see ID3v1Tag()
*/ */
PropertyMap setProperties(const PropertyMap &); bool hasID3v1Tag() const;
/*! /*!
* Returns the APE::Properties for this file. If no audio properties * Returns whether or not the given \a stream can be opened as an APE file.
* were read then this will return a null pointer. *
*/ * \note This method is designed to do a quick check.
virtual Properties *audioProperties() const; * The result may not necessarily be correct.
*/
static bool isSupported(IOStream *stream);
/*! private:
* Saves the file. File(const File &);
* File &operator=(const File &);
* \note According to the official Monkey's Audio SDK, an APE file
* can only have either ID3V1 or APE tags, so a parameter is used here.
*/
virtual bool save();
/*! void read(bool readProperties);
* Returns a pointer to the ID3v1 tag of the file.
*
* If \a create is false (the default) this may return a null pointer
* if there is no valid ID3v1 tag. If \a create is true it will create
* an ID3v1 tag if one does not exist and returns a valid pointer.
*
* \note This may return a valid pointer regardless of whether or not the
* file on disk has an ID3v1 tag. Use hasID3v1Tag() to check if the file
* on disk actually has an ID3v1 tag.
*
* \note The Tag <b>is still</b> owned by the MPEG::File and should not be
* deleted by the user. It will be deleted when the file (object) is
* destroyed.
*
* \see hasID3v1Tag()
*/
ID3v1::Tag *ID3v1Tag(bool create = false);
/*! class FilePrivate;
* Returns a pointer to the APE tag of the file. FilePrivate *d;
* };
* If \a create is false (the default) this may return a null pointer } // namespace APE
* if there is no valid APE tag. If \a create is true it will create } // namespace TagLib
* an APE tag if one does not exist and returns a valid pointer. } // namespace Strawberry_TagLib
*
* \note This may return a valid pointer regardless of whether or not the
* file on disk has an APE tag. Use hasAPETag() to check if the file
* on disk actually has an APE tag.
*
* \note The Tag <b>is still</b> owned by the MPEG::File and should not be
* deleted by the user. It will be deleted when the file (object) is
* destroyed.
*
* \see hasAPETag()
*/
APE::Tag *APETag(bool create = false);
/*!
* This will remove the tags that match the OR-ed together TagTypes from the
* file. By default it removes all tags.
*
* \note This will also invalidate pointers to the tags
* as their memory will be freed.
* \note In order to make the removal permanent save() still needs to be called
*/
void strip(int tags = AllTags);
/*!
* Returns whether or not the file on disk actually has an APE tag.
*
* \see APETag()
*/
bool hasAPETag() const;
/*!
* Returns whether or not the file on disk actually has an ID3v1 tag.
*
* \see ID3v1Tag()
*/
bool hasID3v1Tag() const;
/*!
* Returns whether or not the given \a stream can be opened as an APE
* file.
*
* \note This method is designed to do a quick check. The result may
* not necessarily be correct.
*/
static bool isSupported(IOStream *stream);
private:
File(const File &);
File &operator=(const File &);
void read(bool readProperties);
class FilePrivate;
FilePrivate *d;
};
}
}
}
#endif #endif

View File

@@ -27,24 +27,22 @@
#include <iostream> #include <iostream>
#include <bitset> #include <bitset>
#include <tstring.h> #include "tstring.h"
#include <tdebug.h> #include "tdebug.h"
#include "apefooter.h" #include "apefooter.h"
using namespace Strawberry_TagLib::TagLib; using namespace Strawberry_TagLib::TagLib;
using namespace APE; using namespace APE;
class APE::Footer::FooterPrivate class APE::Footer::FooterPrivate {
{ public:
public: FooterPrivate() : version(0),
FooterPrivate() : footerPresent(true),
version(0), headerPresent(false),
footerPresent(true), isHeader(false),
headerPresent(false), itemCount(0),
isHeader(false), tagSize(0) {}
itemCount(0),
tagSize(0) {}
unsigned int version; unsigned int version;
@@ -61,13 +59,11 @@ public:
// static members // static members
//////////////////////////////////////////////////////////////////////////////// ////////////////////////////////////////////////////////////////////////////////
unsigned int APE::Footer::size() unsigned int APE::Footer::size() {
{
return 32; return 32;
} }
ByteVector APE::Footer::fileIdentifier() ByteVector APE::Footer::fileIdentifier() {
{
return ByteVector("APETAGEX"); return ByteVector("APETAGEX");
} }
@@ -75,119 +71,103 @@ ByteVector APE::Footer::fileIdentifier()
// public members // public members
//////////////////////////////////////////////////////////////////////////////// ////////////////////////////////////////////////////////////////////////////////
APE::Footer::Footer() : APE::Footer::Footer() : d(new FooterPrivate()) {
d(new FooterPrivate())
{
} }
APE::Footer::Footer(const ByteVector &data) : APE::Footer::Footer(const ByteVector &data) : d(new FooterPrivate()) {
d(new FooterPrivate())
{
parse(data); parse(data);
} }
APE::Footer::~Footer() APE::Footer::~Footer() {
{
delete d; delete d;
} }
unsigned int APE::Footer::version() const unsigned int APE::Footer::version() const {
{
return d->version; return d->version;
} }
bool APE::Footer::headerPresent() const bool APE::Footer::headerPresent() const {
{
return d->headerPresent; return d->headerPresent;
} }
bool APE::Footer::footerPresent() const bool APE::Footer::footerPresent() const {
{
return d->footerPresent; return d->footerPresent;
} }
bool APE::Footer::isHeader() const bool APE::Footer::isHeader() const {
{
return d->isHeader; return d->isHeader;
} }
void APE::Footer::setHeaderPresent(bool b) const void APE::Footer::setHeaderPresent(bool b) const {
{
d->headerPresent = b; d->headerPresent = b;
} }
unsigned int APE::Footer::itemCount() const unsigned int APE::Footer::itemCount() const {
{
return d->itemCount; return d->itemCount;
} }
void APE::Footer::setItemCount(unsigned int s) void APE::Footer::setItemCount(unsigned int s) {
{
d->itemCount = s; d->itemCount = s;
} }
unsigned int APE::Footer::tagSize() const unsigned int APE::Footer::tagSize() const {
{
return d->tagSize; return d->tagSize;
} }
unsigned int APE::Footer::completeTagSize() const unsigned int APE::Footer::completeTagSize() const {
{ if (d->headerPresent)
if(d->headerPresent)
return d->tagSize + size(); return d->tagSize + size();
else else
return d->tagSize; return d->tagSize;
} }
void APE::Footer::setTagSize(unsigned int s) void APE::Footer::setTagSize(unsigned int s) {
{
d->tagSize = s; d->tagSize = s;
} }
void APE::Footer::setData(const ByteVector &data) void APE::Footer::setData(const ByteVector &data) {
{
parse(data); parse(data);
} }
ByteVector APE::Footer::renderFooter() const ByteVector APE::Footer::renderFooter() const {
{
return render(false); return render(false);
} }
ByteVector APE::Footer::renderHeader() const ByteVector APE::Footer::renderHeader() const {
{
if(!d->headerPresent) if (!d->headerPresent)
return ByteVector(); return ByteVector();
else else
return render(true); return render(true);
} }
//////////////////////////////////////////////////////////////////////////////// ////////////////////////////////////////////////////////////////////////////////
// protected members // protected members
//////////////////////////////////////////////////////////////////////////////// ////////////////////////////////////////////////////////////////////////////////
void APE::Footer::parse(const ByteVector &data) void APE::Footer::parse(const ByteVector &data) {
{
if(data.size() < size()) if (data.size() < size())
return; return;
// The first eight bytes, data[0..7], are the File Identifier, "APETAGEX". // The first eight bytes, data[0..7], are the File Identifier, "APETAGEX".
// Read the version number // Read the version number
d->version = data.toUInt(8, false); d->version = data.toUInt32LE(8);
// Read the tag size // Read the tag size
d->tagSize = data.toUInt(12, false); d->tagSize = data.toUInt32LE(12);
// Read the item count // Read the item count
d->itemCount = data.toUInt(16, false); d->itemCount = data.toUInt32LE(16);
// Read the flags // Read the flags
std::bitset<32> flags(TAGLIB_CONSTRUCT_BITSET(data.toUInt(20, false))); std::bitset<32> flags(TAGLIB_CONSTRUCT_BITSET(data.toUInt32LE(20)));
d->headerPresent = flags[31]; d->headerPresent = flags[31];
d->footerPresent = !flags[30]; d->footerPresent = !flags[30];
@@ -195,8 +175,8 @@ void APE::Footer::parse(const ByteVector &data)
} }
ByteVector APE::Footer::render(bool isHeader) const ByteVector APE::Footer::render(bool isHeader) const {
{
ByteVector v; ByteVector v;
// add the file identifier -- "APETAGEX" // add the file identifier -- "APETAGEX"
@@ -206,29 +186,30 @@ ByteVector APE::Footer::render(bool isHeader) const
// add the version number -- we always render a 2.000 tag regardless of what // add the version number -- we always render a 2.000 tag regardless of what
// the tag originally was. // the tag originally was.
v.append(ByteVector::fromUInt(2000, false)); v.append(ByteVector::fromUInt32LE(2000));
// add the tag size // add the tag size
v.append(ByteVector::fromUInt(d->tagSize, false)); v.append(ByteVector::fromUInt32LE(d->tagSize));
// add the item count // add the item count
v.append(ByteVector::fromUInt(d->itemCount, false)); v.append(ByteVector::fromUInt32LE(d->itemCount));
// render and add the flags // render and add the flags
std::bitset<32> flags; std::bitset<32> flags;
flags[31] = d->headerPresent; flags[31] = d->headerPresent;
flags[30] = false; // footer is always present flags[30] = false; // footer is always present
flags[29] = isHeader; flags[29] = isHeader;
v.append(ByteVector::fromUInt(flags.to_ulong(), false)); v.append(ByteVector::fromUInt32LE(flags.to_ulong()));
// add the reserved 64bit // add the reserved 64bit
v.append(ByteVector::fromLongLong(0)); v.append(ByteVector::fromUInt64BE(0));
return v; return v;
} }

View File

@@ -32,144 +32,144 @@
namespace Strawberry_TagLib { namespace Strawberry_TagLib {
namespace TagLib { namespace TagLib {
namespace APE { namespace APE {
//! An implementation of APE footers //! An implementation of APE footers
/*! /*!
* This class implements APE footers (and headers). It attempts to follow, both * This class implements APE footers (and headers).
* semantically and programmatically, the structure specified in * It attempts to follow, both semantically and programmatically,
* the APE v2.0 standard. The API is based on the properties of APE footer and * the structure specified in the APE v2.0 standard.
* headers specified there. * The API is based on the properties of APE footer and headers specified there.
*/ *
*/
class TAGLIB_EXPORT Footer class TAGLIB_EXPORT Footer {
{ public:
public: /*!
/*! * Constructs an empty APE footer.
* Constructs an empty APE footer. */
*/ explicit Footer();
Footer();
/*! /*!
* Constructs an APE footer based on \a data. parse() is called * Constructs an APE footer based on \a data. parse() is called immediately.
* immediately. */
*/ explicit Footer(const ByteVector &data);
Footer(const ByteVector &data);
/*! /*!
* Destroys the footer. * Destroys the footer.
*/ */
virtual ~Footer(); virtual ~Footer();
/*! /*!
* Returns the version number. (Note: This is the 1000 or 2000.) * Returns the version number. (Note: This is the 1000 or 2000.)
*/ */
unsigned int version() const; unsigned int version() const;
/*! /*!
* Returns true if a header is present in the tag. * Returns true if a header is present in the tag.
*/ */
bool headerPresent() const; bool headerPresent() const;
/*! /*!
* Returns true if a footer is present in the tag. * Returns true if a footer is present in the tag.
*/ */
bool footerPresent() const; bool footerPresent() const;
/*! /*!
* Returns true this is actually the header. * Returns true this is actually the header.
*/ */
bool isHeader() const; bool isHeader() const;
/*! /*!
* Sets whether the header should be rendered or not * Sets whether the header should be rendered or not
*/ */
void setHeaderPresent(bool b) const; void setHeaderPresent(bool b) const;
/*! /*!
* Returns the number of items in the tag. * Returns the number of items in the tag.
*/ */
unsigned int itemCount() const; unsigned int itemCount() const;
/*! /*!
* Set the item count to \a s. * Set the item count to \a s.
* \see itemCount() * \see itemCount()
*/ */
void setItemCount(unsigned int s); void setItemCount(unsigned int s);
/*! /*!
* Returns the tag size in bytes. This is the size of the frame content and footer. * Returns the tag size in bytes.
* The size of the \e entire tag will be this plus the header size, if present. * This is the size of the frame content and footer.
* * The size of the \e entire tag will be this plus the header size, if present.
* \see completeTagSize() *
*/ * \see completeTagSize()
unsigned int tagSize() const; */
unsigned int tagSize() const;
/*! /*!
* Returns the tag size, including if present, the header * Returns the tag size, including if present, the header
* size. * size.
* *
* \see tagSize() * \see tagSize()
*/ */
unsigned int completeTagSize() const; unsigned int completeTagSize() const;
/*! /*!
* Set the tag size to \a s. * Set the tag size to \a s.
* \see tagSize() * \see tagSize()
*/ */
void setTagSize(unsigned int s); void setTagSize(unsigned int s);
/*! /*!
* Returns the size of the footer. Presently this is always 32 bytes. * Returns the size of the footer. Presently this is always 32 bytes.
*/ */
static unsigned int size(); static unsigned int size();
/*! /*!
* Returns the string used to identify an APE tag inside of a file. * Returns the string used to identify an APE tag inside of a file.
* Presently this is always "APETAGEX". * Presently this is always "APETAGEX".
*/ */
static ByteVector fileIdentifier(); static ByteVector fileIdentifier();
/*! /*!
* Sets the data that will be used as the footer. 32 bytes, * Sets the data that will be used as the footer. 32 bytes,
* starting from \a data will be used. * starting from \a data will be used.
*/ */
void setData(const ByteVector &data); void setData(const ByteVector &data);
/*! /*!
* Renders the footer back to binary format. * Renders the footer back to binary format.
*/ */
ByteVector renderFooter() const; ByteVector renderFooter() const;
/*! /*!
* Renders the header corresponding to the footer. If headerPresent is * Renders the header corresponding to the footer.
* set to false, it returns an empty ByteVector. * If headerPresent is set to false, it returns an empty ByteVector.
*/ */
ByteVector renderHeader() const; ByteVector renderHeader() const;
protected: protected:
/*! /*!
* Called by setData() to parse the footer data. It makes this information * Called by setData() to parse the footer data.
* available through the public API. * It makes this information available through the public API.
*/ */
void parse(const ByteVector &data); void parse(const ByteVector &data);
/*! /*!
* Called by renderFooter and renderHeader * Called by renderFooter and renderHeader
*/ */
ByteVector render(bool isHeader) const; ByteVector render(bool isHeader) const;
private: private:
Footer(const Footer &); Footer(const Footer &);
Footer &operator=(const Footer &); Footer &operator=(const Footer &);
class FooterPrivate; class FooterPrivate;
FooterPrivate *d; FooterPrivate *d;
}; };
} } // namespace APE
} } // namespace TagLib
} } // namespace Strawberry_TagLib
#endif #endif

View File

@@ -23,20 +23,18 @@
* http://www.mozilla.org/MPL/ * * http://www.mozilla.org/MPL/ *
***************************************************************************/ ***************************************************************************/
#include <tbytevectorlist.h> #include <memory>
#include <tdebug.h>
#include "tbytevectorlist.h"
#include "tdebug.h"
#include "apeitem.h" #include "apeitem.h"
using namespace Strawberry_TagLib::TagLib; using namespace Strawberry_TagLib::TagLib;
using namespace APE; using namespace APE;
class APE::Item::ItemPrivate struct ItemData {
{ ItemData() : type(Item::Text), readOnly(false) {}
public:
ItemPrivate() :
type(Text),
readOnly(false) {}
Item::ItemTypes type; Item::ItemTypes type;
String key; String key;
@@ -45,257 +43,229 @@ public:
bool readOnly; bool readOnly;
}; };
class APE::Item::ItemPrivate {
public:
ItemPrivate() : data(new ItemData()) {}
std::shared_ptr<ItemData> data;
};
//////////////////////////////////////////////////////////////////////////////// ////////////////////////////////////////////////////////////////////////////////
// public members // public members
//////////////////////////////////////////////////////////////////////////////// ////////////////////////////////////////////////////////////////////////////////
APE::Item::Item() : APE::Item::Item() : d(new ItemPrivate()) {}
d(new ItemPrivate())
{ APE::Item::Item(const String &key, const String &value) : d(new ItemPrivate()) {
d->data->key = key;
d->data->text.append(value);
} }
APE::Item::Item(const String &key, const String &value) : APE::Item::Item(const String &key, const StringList &values) : d(new ItemPrivate()) {
d(new ItemPrivate()) d->data->key = key;
{ d->data->text = values;
d->key = key;
d->text.append(value);
} }
APE::Item::Item(const String &key, const StringList &values) : APE::Item::Item(const String &key, const ByteVector &value, bool binary) : d(new ItemPrivate()) {
d(new ItemPrivate()) d->data->key = key;
{ if (binary) {
d->key = key; d->data->type = Binary;
d->text = values; d->data->value = value;
}
APE::Item::Item(const String &key, const ByteVector &value, bool binary) :
d(new ItemPrivate())
{
d->key = key;
if(binary) {
d->type = Binary;
d->value = value;
} }
else { else {
d->text.append(value); d->data->text.append(value);
} }
} }
APE::Item::Item(const Item &item) : APE::Item::Item(const Item &item) : d(new ItemPrivate(*item.d)) {}
d(new ItemPrivate(*item.d))
{
}
APE::Item::~Item() APE::Item::~Item() {
{
delete d; delete d;
} }
Item &APE::Item::operator=(const Item &item) Item &APE::Item::operator=(const Item &item) {
{
Item(item).swap(*this); Item(item).swap(*this);
return *this; return *this;
} }
void APE::Item::swap(Item &item) void APE::Item::swap(Item &item) {
{
using std::swap; using std::swap;
swap(d, item.d); swap(d, item.d);
} }
void APE::Item::setReadOnly(bool readOnly) void APE::Item::setReadOnly(bool readOnly) {
{ d->data->readOnly = readOnly;
d->readOnly = readOnly;
} }
bool APE::Item::isReadOnly() const bool APE::Item::isReadOnly() const {
{ return d->data->readOnly;
return d->readOnly;
} }
void APE::Item::setType(APE::Item::ItemTypes val) void APE::Item::setType(APE::Item::ItemTypes val) {
{ d->data->type = val;
d->type = val;
} }
APE::Item::ItemTypes APE::Item::type() const APE::Item::ItemTypes APE::Item::type() const {
{ return d->data->type;
return d->type;
} }
String APE::Item::key() const String APE::Item::key() const {
{ return d->data->key;
return d->key;
} }
ByteVector APE::Item::binaryData() const ByteVector APE::Item::binaryData() const {
{ return d->data->value;
return d->value;
} }
void APE::Item::setBinaryData(const ByteVector &value) void APE::Item::setBinaryData(const ByteVector &value) {
{ d->data->type = Binary;
d->type = Binary; d->data->value = value;
d->value = value; d->data->text.clear();
d->text.clear();
} }
ByteVector APE::Item::value() const void APE::Item::setKey(const String &key) {
{ d->data->key = key;
// This seems incorrect as it won't be actually rendering the value to keep it
// up to date.
return d->value;
} }
void APE::Item::setKey(const String &key) void APE::Item::setValue(const String &value) {
{ d->data->type = Text;
d->key = key; d->data->text = value;
d->data->value.clear();
} }
void APE::Item::setValue(const String &value) void APE::Item::setValues(const StringList &value) {
{ d->data->type = Text;
d->type = Text; d->data->text = value;
d->text = value; d->data->value.clear();
d->value.clear();
} }
void APE::Item::setValues(const StringList &value) void APE::Item::appendValue(const String &value) {
{ d->data->type = Text;
d->type = Text; d->data->text.append(value);
d->text = value; d->data->value.clear();
d->value.clear();
} }
void APE::Item::appendValue(const String &value) void APE::Item::appendValues(const StringList &values) {
{ d->data->type = Text;
d->type = Text; d->data->text.append(values);
d->text.append(value); d->data->value.clear();
d->value.clear();
} }
void APE::Item::appendValues(const StringList &values) int APE::Item::size() const {
{ size_t result = 8 + d->data->key.size() + 1;
d->type = Text; switch (d->data->type) {
d->text.append(values);
d->value.clear();
}
int APE::Item::size() const
{
int result = 8 + d->key.size() + 1;
switch(d->type) {
case Text: case Text:
if(!d->text.isEmpty()) { if (!d->data->text.isEmpty()) {
StringList::ConstIterator it = d->text.begin(); StringList::ConstIterator it = d->data->text.begin();
result += it->data(String::UTF8).size(); result += it->data(String::UTF8).size();
it++; it++;
for(; it != d->text.end(); ++it) for (; it != d->data->text.end(); ++it)
result += 1 + it->data(String::UTF8).size(); result += 1 + it->data(String::UTF8).size();
} }
break; break;
case Binary: case Binary:
case Locator: case Locator:
result += d->value.size(); result += d->data->value.size();
break; break;
} }
return result; return result;
} }
StringList APE::Item::toStringList() const StringList APE::Item::values() const {
{ return d->data->text;
return d->text;
} }
StringList APE::Item::values() const String APE::Item::toString() const {
{ if (d->data->type == Text && !isEmpty())
return d->text; return d->data->text.front();
}
String APE::Item::toString() const
{
if(d->type == Text && !isEmpty())
return d->text.front();
else else
return String(); return String();
} }
bool APE::Item::isEmpty() const bool APE::Item::isEmpty() const {
{ switch (d->data->type) {
switch(d->type) {
case Text: case Text:
if(d->text.isEmpty()) if (d->data->text.isEmpty())
return true; return true;
if(d->text.size() == 1 && d->text.front().isEmpty()) if (d->data->text.size() == 1 && d->data->text.front().isEmpty())
return true; return true;
return false; return false;
case Binary: case Binary:
case Locator: case Locator:
return d->value.isEmpty(); return d->data->value.isEmpty();
default: default:
return false; return false;
} }
} }
void APE::Item::parse(const ByteVector &data) void APE::Item::parse(const ByteVector &data) {
{
// 11 bytes is the minimum size for an APE item // 11 bytes is the minimum size for an APE item
if(data.size() < 11) { if (data.size() < 11) {
debug("APE::Item::parse() -- no data in item"); debug("APE::Item::parse() -- no data in item");
return; return;
} }
const unsigned int valueLength = data.toUInt(0, false); const unsigned int valueLength = data.toUInt32LE(0);
const unsigned int flags = data.toUInt(4, false); const unsigned int flags = data.toUInt32LE(4);
// An item key can contain ASCII characters from 0x20 up to 0x7E, not UTF-8. // An item key can contain ASCII characters from 0x20 up to 0x7E, not UTF-8.
// We assume that the validity of the given key has been checked. // We assume that the validity of the given key has been checked.
d->key = String(&data[8], String::Latin1); d->data->key = String(&data[8], String::Latin1);
const ByteVector value = data.mid(8 + d->key.size() + 1, valueLength); const ByteVector value = data.mid(8 + d->data->key.size() + 1, valueLength);
setReadOnly(flags & 1); setReadOnly(flags & 1);
setType(ItemTypes((flags >> 1) & 3)); setType(ItemTypes((flags >> 1) & 3));
if(Text == d->type) if (Text == d->data->type)
d->text = StringList(ByteVectorList::split(value, '\0'), String::UTF8); d->data->text = StringList(ByteVectorList::split(value, '\0'), String::UTF8);
else else
d->value = value; d->data->value = value;
} }
ByteVector APE::Item::render() const ByteVector APE::Item::render() const {
{
ByteVector data; ByteVector data;
unsigned int flags = ((d->readOnly) ? 1 : 0) | (d->type << 1); unsigned int flags = ((d->data->readOnly) ? 1 : 0) | (d->data->type << 1);
ByteVector value; ByteVector value;
if(isEmpty()) if (isEmpty())
return data; return data;
if(d->type == Text) { if (d->data->type == Text) {
StringList::ConstIterator it = d->text.begin(); StringList::ConstIterator it = d->data->text.begin();
value.append(it->data(String::UTF8)); value.append(it->data(String::UTF8));
it++; it++;
for(; it != d->text.end(); ++it) { for (; it != d->data->text.end(); ++it) {
value.append('\0'); value.append('\0');
value.append(it->data(String::UTF8)); value.append(it->data(String::UTF8));
} }
d->value = value; d->data->value = value;
} }
else else
value.append(d->value); value.append(d->data->value);
data.append(ByteVector::fromUInt(value.size(), false)); data.append(ByteVector::fromUInt32LE(value.size()));
data.append(ByteVector::fromUInt(flags, false)); data.append(ByteVector::fromUInt32LE(flags));
data.append(d->key.data(String::Latin1)); data.append(d->data->key.data(String::Latin1));
data.append(ByteVector('\0')); data.append(ByteVector('\0'));
data.append(value); data.append(value);
return data; return data;
} }

View File

@@ -32,195 +32,176 @@
namespace Strawberry_TagLib { namespace Strawberry_TagLib {
namespace TagLib { namespace TagLib {
namespace APE {
namespace APE { //! An implementation of APE-items
//! An implementation of APE-items /*!
* This class provides the features of items in the APEv2 standard.
*/
class TAGLIB_EXPORT Item {
public:
/*!
* Enum of types an Item can have. The value of 3 is reserved.
*/
enum ItemTypes {
//! Item contains text information coded in UTF-8
Text = 0,
//! Item contains binary information
Binary = 1,
//! Item is a locator of external stored information
Locator = 2
};
/*!
* Constructs an empty item.
*/
explicit Item();
/*! /*!
* This class provides the features of items in the APEv2 standard. * Constructs a text item with \a key and \a values.
*/ */
class TAGLIB_EXPORT Item explicit Item(const String &key, const String &values);
{
public:
/*!
* Enum of types an Item can have. The value of 3 is reserved.
*/
enum ItemTypes {
//! Item contains text information coded in UTF-8
Text = 0,
//! Item contains binary information
Binary = 1,
//! Item is a locator of external stored information
Locator = 2
};
/*!
* Constructs an empty item.
*/
Item();
/*! /*!
* Constructs a text item with \a key and \a value. * Constructs a text item with \a key and \a values.
*/ */
// BIC: Remove this, StringList has a constructor from a single string explicit Item(const String &key, const StringList &values);
Item(const String &key, const String &value);
/*! /*!
* Constructs a text item with \a key and \a values. * Constructs an item with \a key and \a value.
*/ * If \a binary is true a Binary item will be created, otherwise \a value will be interpreted as text
Item(const String &key, const StringList &values); */
explicit Item(const String &key, const ByteVector &value, bool binary);
/*! /*!
* Constructs an item with \a key and \a value. * Construct an item as a copy of \a item.
* If \a binary is true a Binary item will be created, otherwise \a value will be interpreted as text */
*/ Item(const Item &item);
Item(const String &key, const ByteVector &value, bool binary);
/*! /*!
* Construct an item as a copy of \a item. * Destroys the item.
*/ */
Item(const Item &item); virtual ~Item();
/*! /*!
* Destroys the item. * Copies the contents of \a item into this item.
*/ */
virtual ~Item(); Item &operator=(const Item &item);
/*! /*!
* Copies the contents of \a item into this item. * Exchanges the content of this item by the content of \a item.
*/ */
Item &operator=(const Item &item); void swap(Item &item);
/*! /*!
* Exchanges the content of this item by the content of \a item. * Returns the key.
*/ */
void swap(Item &item); String key() const;
/*! /*!
* Returns the key. * Returns the binary value.
*/ * If the item type is not \a Binary, always returns an empty ByteVector.
String key() const; */
ByteVector binaryData() const;
/*! /*!
* Returns the binary value. * Set the binary value to \a value
* If the item type is not \a Binary, always returns an empty ByteVector. * The item's type will also be set to \a Binary
*/ */
ByteVector binaryData() const; void setBinaryData(const ByteVector &value);
/*! /*!
* Set the binary value to \a value * Sets the key for the item to \a key.
* The item's type will also be set to \a Binary */
*/ void setKey(const String &key);
void setBinaryData(const ByteVector &value);
#ifndef DO_NOT_DOCUMENT /*!
/* Remove in next binary incompatible release */ * Sets the text value of the item to \a value and clears any previous contents.
ByteVector value() const; *
#endif * \see toString()
*/
void setValue(const String &value);
/*! /*!
* Sets the key for the item to \a key. * Sets the text value of the item to the list of values in \a value and clears any previous contents.
*/ *
void setKey(const String &key); * \see toStringList()
*/
void setValues(const StringList &values);
/*! /*!
* Sets the text value of the item to \a value and clears any previous contents. * Appends \a value to create (or extend) the current list of text values.
* *
* \see toString() * \see toString()
*/ */
void setValue(const String &value); void appendValue(const String &value);
/*! /*!
* Sets the text value of the item to the list of values in \a value and clears * Appends \a values to extend the current list of text values.
* any previous contents. *
* * \see toStringList()
* \see toStringList() */
*/ void appendValues(const StringList &values);
void setValues(const StringList &values);
/*! /*!
* Appends \a value to create (or extend) the current list of text values. * Returns the size of the full item.
* */
* \see toString() int size() const;
*/
void appendValue(const String &value);
/*! /*!
* Appends \a values to extend the current list of text values. * Returns the value as a single string. In case of multiple strings, the first is returned.
* * If the data type is not \a Text, always returns an empty String.
* \see toStringList() */
*/ String toString() const;
void appendValues(const StringList &values);
/*! /*!
* Returns the size of the full item. * Returns the list of text values. If the data type is not \a Text, always returns an empty StringList.
*/ */
int size() const; StringList values() const;
/*! /*!
* Returns the value as a single string. In case of multiple strings, * Render the item to a ByteVector.
* the first is returned. If the data type is not \a Text, always returns */
* an empty String. ByteVector render() const;
*/
String toString() const;
#ifndef DO_NOT_DOCUMENT /*!
/* Remove in next binary incompatible release */ * Parse the item from the ByteVector \a data.
StringList toStringList() const; */
#endif void parse(const ByteVector &data);
/*! /*!
* Returns the list of text values. If the data type is not \a Text, always * Set the item to read-only.
* returns an empty StringList. */
*/ void setReadOnly(bool readOnly);
StringList values() const;
/*! /*!
* Render the item to a ByteVector. * Return true if the item is read-only.
*/ */
ByteVector render() const; bool isReadOnly() const;
/*! /*!
* Parse the item from the ByteVector \a data. * Sets the type of the item to \a type.
*/ *
void parse(const ByteVector& data); * \see ItemTypes
*/
void setType(ItemTypes type);
/*! /*!
* Set the item to read-only. * Returns the type of the item.
*/ */
void setReadOnly(bool readOnly); ItemTypes type() const;
/*! /*!
* Return true if the item is read-only. * Returns if the item has any real content.
*/ */
bool isReadOnly() const; bool isEmpty() const;
/*! private:
* Sets the type of the item to \a type. class ItemPrivate;
* ItemPrivate *d;
* \see ItemTypes };
*/ } // namespace APE
void setType(ItemTypes type); } // namespace TagLib
} // namespace Strawberry_TagLib
/*!
* Returns the type of the item.
*/
ItemTypes type() const;
/*!
* Returns if the item has any real content.
*/
bool isEmpty() const;
private:
class ItemPrivate;
ItemPrivate *d;
};
}
}
}
#endif #endif

View File

@@ -27,9 +27,9 @@
* http://www.mozilla.org/MPL/ * * http://www.mozilla.org/MPL/ *
***************************************************************************/ ***************************************************************************/
#include <tstring.h> #include "tstring.h"
#include <tdebug.h> #include "tdebug.h"
#include <bitset>
#include "id3v2tag.h" #include "id3v2tag.h"
#include "apeproperties.h" #include "apeproperties.h"
#include "apefile.h" #include "apefile.h"
@@ -38,17 +38,15 @@
using namespace Strawberry_TagLib::TagLib; using namespace Strawberry_TagLib::TagLib;
class APE::Properties::PropertiesPrivate class APE::AudioProperties::AudioPropertiesPrivate {
{ public:
public: explicit AudioPropertiesPrivate() : length(0),
PropertiesPrivate() : bitrate(0),
length(0), sampleRate(0),
bitrate(0), channels(0),
sampleRate(0), version(0),
channels(0), bitsPerSample(0),
version(0), sampleFrames(0) {}
bitsPerSample(0),
sampleFrames(0) {}
int length; int length;
int bitrate; int bitrate;
@@ -63,67 +61,43 @@ public:
// public members // public members
//////////////////////////////////////////////////////////////////////////////// ////////////////////////////////////////////////////////////////////////////////
APE::Properties::Properties(File *, ReadStyle style) : APE::AudioProperties::AudioProperties(File *file, long long streamLength, ReadStyle) : Strawberry_TagLib::TagLib::AudioProperties(), d(new AudioPropertiesPrivate()) {
AudioProperties(style),
d(new PropertiesPrivate())
{
debug("APE::Properties::Properties() -- This constructor is no longer used.");
}
APE::Properties::Properties(File *file, long streamLength, ReadStyle style) :
AudioProperties(style),
d(new PropertiesPrivate())
{
read(file, streamLength); read(file, streamLength);
} }
APE::Properties::~Properties() APE::AudioProperties::~AudioProperties() {
{
delete d; delete d;
} }
int APE::Properties::length() const int APE::AudioProperties::lengthInSeconds() const {
{
return lengthInSeconds();
}
int APE::Properties::lengthInSeconds() const
{
return d->length / 1000; return d->length / 1000;
} }
int APE::Properties::lengthInMilliseconds() const int APE::AudioProperties::lengthInMilliseconds() const {
{
return d->length; return d->length;
} }
int APE::Properties::bitrate() const int APE::AudioProperties::bitrate() const {
{
return d->bitrate; return d->bitrate;
} }
int APE::Properties::sampleRate() const int APE::AudioProperties::sampleRate() const {
{
return d->sampleRate; return d->sampleRate;
} }
int APE::Properties::channels() const int APE::AudioProperties::channels() const {
{
return d->channels; return d->channels;
} }
int APE::Properties::version() const int APE::AudioProperties::version() const {
{
return d->version; return d->version;
} }
int APE::Properties::bitsPerSample() const int APE::AudioProperties::bitsPerSample() const {
{
return d->bitsPerSample; return d->bitsPerSample;
} }
unsigned int APE::Properties::sampleFrames() const unsigned int APE::AudioProperties::sampleFrames() const {
{
return d->sampleFrames; return d->sampleFrames;
} }
@@ -131,122 +105,123 @@ unsigned int APE::Properties::sampleFrames() const
// private members // private members
//////////////////////////////////////////////////////////////////////////////// ////////////////////////////////////////////////////////////////////////////////
namespace namespace {
{ int headerVersion(const ByteVector &header) {
int headerVersion(const ByteVector &header) if (header.size() < 6 || !header.startsWith("MAC "))
{ return -1;
if(header.size() < 6 || !header.startsWith("MAC "))
return -1;
return header.toUShort(4, false); return header.toUInt16LE(4);
}
} }
} // namespace
void APE::AudioProperties::read(File *file, long long streamLength) {
void APE::Properties::read(File *file, long streamLength)
{
// First, we assume that the file pointer is set at the first descriptor. // First, we assume that the file pointer is set at the first descriptor.
long offset = file->tell(); long long offset = file->tell();
int version = headerVersion(file->readBlock(6)); int version = headerVersion(file->readBlock(6));
// Next, we look for the descriptor. // Next, we look for the descriptor.
if(version < 0) { if (version < 0) {
offset = file->find("MAC ", offset); offset = file->find("MAC ", offset);
file->seek(offset); file->seek(offset);
version = headerVersion(file->readBlock(6)); version = headerVersion(file->readBlock(6));
} }
if(version < 0) { if (version < 0) {
debug("APE::Properties::read() -- APE descriptor not found"); debug("APE::AudioProperties::read() -- APE descriptor not found");
return; return;
} }
d->version = version; d->version = version;
if(d->version >= 3980) if (d->version >= 3980)
analyzeCurrent(file); analyzeCurrent(file);
else else
analyzeOld(file); analyzeOld(file);
if(d->sampleFrames > 0 && d->sampleRate > 0) { if (d->sampleFrames > 0 && d->sampleRate > 0) {
const double length = d->sampleFrames * 1000.0 / d->sampleRate; const double length = d->sampleFrames * 1000.0 / d->sampleRate;
d->length = static_cast<int>(length + 0.5); d->length = static_cast<int>(length + 0.5);
d->bitrate = static_cast<int>(streamLength * 8.0 / length + 0.5); d->bitrate = static_cast<int>(streamLength * 8.0 / length + 0.5);
} }
} }
void APE::Properties::analyzeCurrent(File *file) void APE::AudioProperties::analyzeCurrent(File *file) {
{
// Read the descriptor // Read the descriptor
file->seek(2, File::Current); file->seek(2, File::Current);
const ByteVector descriptor = file->readBlock(44); const ByteVector descriptor = file->readBlock(44);
if(descriptor.size() < 44) { if (descriptor.size() < 44) {
debug("APE::Properties::analyzeCurrent() -- descriptor is too short."); debug("APE::AudioProperties::analyzeCurrent() -- descriptor is too short.");
return; return;
} }
const unsigned int descriptorBytes = descriptor.toUInt(0, false); const unsigned int descriptorBytes = descriptor.toUInt32LE(0);
if((descriptorBytes - 52) > 0) if ((descriptorBytes - 52) > 0)
file->seek(descriptorBytes - 52, File::Current); file->seek(descriptorBytes - 52, File::Current);
// Read the header // Read the header
const ByteVector header = file->readBlock(24); const ByteVector header = file->readBlock(24);
if(header.size() < 24) { if (header.size() < 24) {
debug("APE::Properties::analyzeCurrent() -- MAC header is too short."); debug("APE::AudioProperties::analyzeCurrent() -- MAC header is too short.");
return; return;
} }
// Get the APE info // Get the APE info
d->channels = header.toShort(18, false); d->channels = header.toUInt16LE(18);
d->sampleRate = header.toUInt(20, false); d->sampleRate = header.toUInt32LE(20);
d->bitsPerSample = header.toShort(16, false); d->bitsPerSample = header.toUInt16LE(16);
const unsigned int totalFrames = header.toUInt(12, false); const unsigned int totalFrames = header.toUInt32LE(12);
if(totalFrames == 0) if (totalFrames == 0)
return; return;
const unsigned int blocksPerFrame = header.toUInt(4, false); const unsigned int blocksPerFrame = header.toUInt32LE(4);
const unsigned int finalFrameBlocks = header.toUInt(8, false); const unsigned int finalFrameBlocks = header.toUInt32LE(8);
d->sampleFrames = (totalFrames - 1) * blocksPerFrame + finalFrameBlocks; d->sampleFrames = (totalFrames - 1) * blocksPerFrame + finalFrameBlocks;
} }
void APE::Properties::analyzeOld(File *file) void APE::AudioProperties::analyzeOld(File *file) {
{
const ByteVector header = file->readBlock(26); const ByteVector header = file->readBlock(26);
if(header.size() < 26) { if (header.size() < 26) {
debug("APE::Properties::analyzeOld() -- MAC header is too short."); debug("APE::AudioProperties::analyzeOld() -- MAC header is too short.");
return; return;
} }
const unsigned int totalFrames = header.toUInt(18, false); const unsigned int totalFrames = header.toUInt32LE(18);
// Fail on 0 length APE files (catches non-finalized APE files) // Fail on 0 length APE files (catches non-finalized APE files)
if(totalFrames == 0) if (totalFrames == 0)
return; return;
const short compressionLevel = header.toShort(0, false); const short compressionLevel = header.toUInt32LE(0);
unsigned int blocksPerFrame; unsigned int blocksPerFrame;
if(d->version >= 3950) if (d->version >= 3950)
blocksPerFrame = 73728 * 4; blocksPerFrame = 73728 * 4;
else if(d->version >= 3900 || (d->version >= 3800 && compressionLevel == 4000)) else if (d->version >= 3900 || (d->version >= 3800 && compressionLevel == 4000))
blocksPerFrame = 73728; blocksPerFrame = 73728;
else else
blocksPerFrame = 9216; blocksPerFrame = 9216;
// Get the APE info // Get the APE info
d->channels = header.toShort(4, false); d->channels = header.toUInt16LE(4);
d->sampleRate = header.toUInt(6, false); d->sampleRate = header.toUInt32LE(6);
const unsigned int finalFrameBlocks = header.toUInt(22, false); const unsigned int finalFrameBlocks = header.toUInt32LE(22);
d->sampleFrames = (totalFrames - 1) * blocksPerFrame + finalFrameBlocks; d->sampleFrames = (totalFrames - 1) * blocksPerFrame + finalFrameBlocks;
// Get the bit depth from the RIFF-fmt chunk. // Get the bit depth from the RIFF-fmt chunk.
file->seek(16, File::Current); file->seek(16, File::Current);
const ByteVector fmt = file->readBlock(28); const ByteVector fmt = file->readBlock(28);
if(fmt.size() < 28 || !fmt.startsWith("WAVEfmt ")) { if (fmt.size() < 28 || !fmt.startsWith("WAVEfmt ")) {
debug("APE::Properties::analyzeOld() -- fmt header is too short."); debug("APE::AudioProperties::analyzeOld() -- fmt header is too short.");
return; return;
} }
d->bitsPerSample = fmt.toShort(26, false); d->bitsPerSample = fmt.toUInt16LE(26);
} }

View File

@@ -35,111 +35,84 @@
namespace Strawberry_TagLib { namespace Strawberry_TagLib {
namespace TagLib { namespace TagLib {
namespace APE {
namespace APE { class File;
class File; //! An implementation of audio property reading for APE
//! An implementation of audio property reading for APE /*!
* This reads the data from an APE stream found in the AudioProperties API.
*/
/*! class TAGLIB_EXPORT AudioProperties : public Strawberry_TagLib::TagLib::AudioProperties {
* This reads the data from an APE stream found in the AudioProperties public:
* API.
*/
class TAGLIB_EXPORT Properties : public AudioProperties /*!
{ * Create an instance of APE::AudioProperties with the data read from the APE::File \a file.
public: */
/*! explicit AudioProperties(File *file, long long streamLength, ReadStyle style = Average);
* Create an instance of APE::Properties with the data read from the
* APE::File \a file.
*
* \deprecated
*/
Properties(File *file, ReadStyle style = Average);
/*! /*!
* Create an instance of APE::Properties with the data read from the * Destroys this APE::AudioProperties instance.
* APE::File \a file. */
*/ ~AudioProperties() override;
Properties(File *file, long streamLength, ReadStyle style = Average);
/*! /*!
* Destroys this APE::Properties instance. * Returns the length of the file in seconds. The length is rounded down to the nearest whole second.
*/ *
virtual ~Properties(); * \see lengthInMilliseconds()
*/
int lengthInSeconds() const override;
/*! /*!
* Returns the length of the file in seconds. The length is rounded down to * Returns the length of the file in milliseconds.
* the nearest whole second. *
* * \see lengthInSeconds()
* \note This method is just an alias of lengthInSeconds(). */
* int lengthInMilliseconds() const override;
* \deprecated
*/
virtual int length() const;
/*! /*!
* Returns the length of the file in seconds. The length is rounded down to * Returns the average bit rate of the file in kb/s.
* the nearest whole second. */
* int bitrate() const override;
* \see lengthInMilliseconds()
*/
// BIC: make virtual
int lengthInSeconds() const;
/*! /*!
* Returns the length of the file in milliseconds. * Returns the sample rate in Hz.
* */
* \see lengthInSeconds() int sampleRate() const override;
*/
// BIC: make virtual
int lengthInMilliseconds() const;
/*! /*!
* Returns the average bit rate of the file in kb/s. * Returns the number of audio channels.
*/ */
virtual int bitrate() const; int channels() const override;
/*! /*!
* Returns the sample rate in Hz. * Returns the number of bits per audio sample.
*/ */
virtual int sampleRate() const; int bitsPerSample() const;
/*! /*!
* Returns the number of audio channels. * Returns the total number of audio samples in file.
*/ */
virtual int channels() const; unsigned int sampleFrames() const;
/*! /*!
* Returns the number of bits per audio sample. * Returns APE version.
*/ */
int bitsPerSample() const; int version() const;
/*! private:
* Returns the total number of audio samples in file. void read(File *file, long long streamLength);
*/
unsigned int sampleFrames() const;
/*! void analyzeCurrent(File *file);
* Returns APE version. void analyzeOld(File *file);
*/
int version() const;
private: class AudioPropertiesPrivate;
Properties(const Properties &); AudioPropertiesPrivate *d;
Properties &operator=(const Properties &); };
} // namespace APE
void read(File *file, long streamLength); } // namespace TagLib
} // namespace Strawberry_TagLib
void analyzeCurrent(File *file);
void analyzeOld(File *file);
class PropertiesPrivate;
PropertiesPrivate *d;
};
}
}
}
#endif #endif

View File

@@ -28,15 +28,16 @@
// it considers specializations with and without class types // it considers specializations with and without class types
// to be different; this define forces Map to use only the // to be different; this define forces Map to use only the
// specialization with the class keyword. // specialization with the class keyword.
#define WANT_CLASS_INSTANTIATION_OF_MAP (1) # define WANT_CLASS_INSTANTIATION_OF_MAP (1)
#endif #endif
#include <tfile.h> #include "tfile.h"
#include <tstring.h> #include "tstring.h"
#include <tmap.h> #include "tmap.h"
#include <tpropertymap.h> #include "tpicturemap.h"
#include <tdebug.h> #include "tpropertymap.h"
#include <tutils.h> #include "tdebug.h"
#include "tutils.h"
#include "apetag.h" #include "apetag.h"
#include "apefooter.h" #include "apefooter.h"
@@ -45,42 +46,38 @@
using namespace Strawberry_TagLib::TagLib; using namespace Strawberry_TagLib::TagLib;
using namespace APE; using namespace APE;
namespace namespace {
{ const unsigned int MinKeyLength = 2;
const unsigned int MinKeyLength = 2; const unsigned int MaxKeyLength = 255;
const unsigned int MaxKeyLength = 255;
bool isKeyValid(const ByteVector &key) bool isKeyValid(const ByteVector &key) {
{
const char *invalidKeys[] = { "ID3", "TAG", "OGGS", "MP+", 0 };
// only allow printable ASCII including space (32..126) const char *invalidKeys[] = { "ID3", "TAG", "OGGS", "MP+", nullptr };
for(ByteVector::ConstIterator it = key.begin(); it != key.end(); ++it) { // only allow printable ASCII including space (32..126)
const int c = static_cast<unsigned char>(*it);
if(c < 32 || c > 126)
return false;
}
const String upperKey = String(key).upper(); for (ByteVector::ConstIterator it = key.begin(); it != key.end(); ++it) {
for(size_t i = 0; invalidKeys[i] != 0; ++i) { const int c = static_cast<unsigned char>(*it);
if(upperKey == invalidKeys[i]) if (c < 32 || c > 126)
return false; return false;
}
return true;
} }
}
class APE::Tag::TagPrivate const String upperKey = String(key).upper();
{ for (size_t i = 0; invalidKeys[i] != nullptr; ++i) {
public: if (upperKey == invalidKeys[i])
TagPrivate() : return false;
file(0), }
footerLocation(0) {}
return true;
}
} // namespace
class APE::Tag::TagPrivate {
public:
TagPrivate() : file(nullptr), footerLocation(0) {}
File *file; File *file;
long footerLocation; long long footerLocation;
Footer footer; Footer footer;
ItemListMap itemListMap; ItemListMap itemListMap;
@@ -90,172 +87,257 @@ public:
// public methods // public methods
//////////////////////////////////////////////////////////////////////////////// ////////////////////////////////////////////////////////////////////////////////
APE::Tag::Tag() : APE::Tag::Tag() : d(new TagPrivate()) {}
Strawberry_TagLib::TagLib::Tag(),
d(new TagPrivate()) APE::Tag::Tag(Strawberry_TagLib::TagLib::File *file, long long footerLocation) : d(new TagPrivate()) {
{
}
APE::Tag::Tag(Strawberry_TagLib::TagLib::File *file, long footerLocation) :
Strawberry_TagLib::TagLib::Tag(),
d(new TagPrivate())
{
d->file = file; d->file = file;
d->footerLocation = footerLocation; d->footerLocation = footerLocation;
read(); read();
} }
APE::Tag::~Tag() APE::Tag::~Tag() {
{
delete d; delete d;
} }
ByteVector APE::Tag::fileIdentifier() ByteVector APE::Tag::fileIdentifier() {
{
return ByteVector::fromCString("APETAGEX"); return ByteVector::fromCString("APETAGEX");
} }
String APE::Tag::title() const String APE::Tag::title() const {
{
if(d->itemListMap["TITLE"].isEmpty()) if (d->itemListMap["TITLE"].isEmpty())
return String(); return String();
return d->itemListMap["TITLE"].values().toString(); return d->itemListMap["TITLE"].values().toString();
} }
String APE::Tag::artist() const String APE::Tag::artist() const {
{
if(d->itemListMap["ARTIST"].isEmpty()) if (d->itemListMap["ARTIST"].isEmpty())
return String(); return String();
return d->itemListMap["ARTIST"].values().toString(); return d->itemListMap["ARTIST"].values().toString();
} }
String APE::Tag::album() const String APE::Tag::album() const {
{
if(d->itemListMap["ALBUM"].isEmpty()) if (d->itemListMap["ALBUM"].isEmpty())
return String(); return String();
return d->itemListMap["ALBUM"].values().toString(); return d->itemListMap["ALBUM"].values().toString();
} }
String APE::Tag::comment() const String APE::Tag::comment() const {
{
if(d->itemListMap["COMMENT"].isEmpty()) if (d->itemListMap["COMMENT"].isEmpty())
return String(); return String();
return d->itemListMap["COMMENT"].values().toString(); return d->itemListMap["COMMENT"].values().toString();
} }
String APE::Tag::genre() const String APE::Tag::genre() const {
{
if(d->itemListMap["GENRE"].isEmpty()) if (d->itemListMap["GENRE"].isEmpty())
return String(); return String();
return d->itemListMap["GENRE"].values().toString(); return d->itemListMap["GENRE"].values().toString();
} }
unsigned int APE::Tag::year() const unsigned int APE::Tag::year() const {
{
if(d->itemListMap["YEAR"].isEmpty()) if (d->itemListMap["YEAR"].isEmpty())
return 0; return 0;
return d->itemListMap["YEAR"].toString().toInt(); return d->itemListMap["YEAR"].toString().toInt();
} }
unsigned int APE::Tag::track() const unsigned int APE::Tag::track() const {
{
if(d->itemListMap["TRACK"].isEmpty()) if (d->itemListMap["TRACK"].isEmpty())
return 0; return 0;
return d->itemListMap["TRACK"].toString().toInt(); return d->itemListMap["TRACK"].toString().toInt();
} }
void APE::Tag::setTitle(const String &s) Strawberry_TagLib::TagLib::PictureMap APE::Tag::pictures() const {
{
PictureMap map;
if (d->itemListMap.contains(FRONT_COVER)) {
Item front = d->itemListMap[FRONT_COVER];
if (Item::Binary == front.type()) {
ByteVector picture = front.binaryData();
const size_t index = picture.find('\0');
if (index < picture.size()) {
ByteVector desc = picture.mid(0, index + 1);
String mime = "image/jpeg";
ByteVector data = picture.mid(index + 1);
Picture p(data, Picture::FrontCover, mime, desc);
map.insert(p);
}
}
}
if (d->itemListMap.contains(BACK_COVER)) {
Item back = d->itemListMap[BACK_COVER];
if (Item::Binary == back.type()) {
ByteVector picture = back.binaryData();
const size_t index = picture.find('\0');
if (index < picture.size()) {
ByteVector desc = picture.mid(0, index + 1);
String mime = "image/jpeg";
ByteVector data = picture.mid(index + 1);
Picture p(data, Picture::BackCover, mime, desc);
map.insert(p);
}
}
}
return PictureMap(map);
}
void APE::Tag::setTitle(const String &s) {
addValue("TITLE", s, true); addValue("TITLE", s, true);
} }
void APE::Tag::setArtist(const String &s) void APE::Tag::setArtist(const String &s) {
{
addValue("ARTIST", s, true); addValue("ARTIST", s, true);
} }
void APE::Tag::setAlbum(const String &s) void APE::Tag::setAlbum(const String &s) {
{
addValue("ALBUM", s, true); addValue("ALBUM", s, true);
} }
void APE::Tag::setComment(const String &s) void APE::Tag::setComment(const String &s) {
{
addValue("COMMENT", s, true); addValue("COMMENT", s, true);
} }
void APE::Tag::setGenre(const String &s) void APE::Tag::setGenre(const String &s) {
{
addValue("GENRE", s, true); addValue("GENRE", s, true);
} }
void APE::Tag::setYear(unsigned int i) void APE::Tag::setYear(unsigned int i) {
{
if(i == 0) if (i == 0)
removeItem("YEAR"); removeItem("YEAR");
else else
addValue("YEAR", String::number(i), true); addValue("YEAR", String::number(i), true);
} }
void APE::Tag::setTrack(unsigned int i) void APE::Tag::setTrack(unsigned int i) {
{
if(i == 0) if (i == 0)
removeItem("TRACK"); removeItem("TRACK");
else else
addValue("TRACK", String::number(i), true); addValue("TRACK", String::number(i), true);
} }
namespace void APE::Tag::setPictures(const PictureMap &l) {
{
// conversions of tag keys between what we use in PropertyMap and what's usual removeItem(FRONT_COVER);
// for APE tags removeItem(BACK_COVER);
// usual, APE
const char *keyConversions[][2] = {{"TRACKNUMBER", "TRACK" }, for (PictureMap::ConstIterator pictureMapIt = l.begin(); pictureMapIt != l.end(); ++pictureMapIt) {
{"DATE", "YEAR" }, Picture::Type type = pictureMapIt->first;
{"ALBUMARTIST", "ALBUM ARTIST"}, if (Picture::FrontCover != type && Picture::BackCover != type) {
{"DISCNUMBER", "DISC" }, std::cout << "APE: Trying to add a picture with wrong type" << std::endl;
{"REMIXER", "MIXARTIST" }}; continue;
const size_t keyConversionsSize = sizeof(keyConversions) / sizeof(keyConversions[0]); }
const char *id;
switch (type) {
case Picture::FrontCover:
id = FRONT_COVER;
break;
case Picture::BackCover:
id = BACK_COVER;
break;
default:
id = FRONT_COVER;
break;
}
PictureList list = pictureMapIt->second;
for (PictureList::ConstIterator pictureListIt = list.begin(); pictureListIt != list.end(); ++pictureListIt) {
Picture picture = *pictureListIt;
if (d->itemListMap.contains(id)) {
std::cout << "APE: Already added a picture of type "
<< id
<< " '"
<< picture.description()
<< "' "
<< "and next are being ignored"
<< std::endl;
break;
}
ByteVector data = picture.description().data(String::Latin1).append('\0').append(picture.data());
Item item;
item.setKey(id);
item.setType(Item::Binary);
item.setBinaryData(data);
setItem(item.key(), item);
}
}
} }
PropertyMap APE::Tag::properties() const namespace {
{ // conversions of tag keys between what we use in PropertyMap and what's usual
// for APE tags
// usual, APE
const char *keyConversions[][2] = { { "TRACKNUMBER", "TRACK" },
{ "DATE", "YEAR" },
{ "ALBUMARTIST", "ALBUM ARTIST" },
{ "DISCNUMBER", "DISC" },
{ "REMIXER", "MIXARTIST" } };
const size_t keyConversionsSize = sizeof(keyConversions) / sizeof(keyConversions[0]);
} // namespace
PropertyMap APE::Tag::properties() const {
PropertyMap properties; PropertyMap properties;
ItemListMap::ConstIterator it = itemListMap().begin(); ItemListMap::ConstIterator it = itemListMap().begin();
for(; it != itemListMap().end(); ++it) { for (; it != itemListMap().end(); ++it) {
String tagName = it->first.upper(); String tagName = it->first.upper();
// if the item is Binary or Locator, or if the key is an invalid string, // if the item is Binary or Locator, or if the key is an invalid string,
// add to unsupportedData // add to unsupportedData
if(it->second.type() != Item::Text || tagName.isEmpty()) { if (it->second.type() != Item::Text || tagName.isEmpty()) {
properties.unsupportedData().append(it->first); properties.unsupportedData().append(it->first);
} }
else { else {
// Some tags need to be handled specially // Some tags need to be handled specially
for(size_t i = 0; i < keyConversionsSize; ++i) { for (size_t i = 0; i < keyConversionsSize; ++i) {
if(tagName == keyConversions[i][1]) if (tagName == keyConversions[i][1])
tagName = keyConversions[i][0]; tagName = keyConversions[i][0];
} }
properties[tagName].append(it->second.toStringList()); properties[tagName].append(it->second.values());
} }
} }
return properties; return properties;
} }
void APE::Tag::removeUnsupportedProperties(const StringList &properties) void APE::Tag::removeUnsupportedProperties(const StringList &properties) {
{
StringList::ConstIterator it = properties.begin(); StringList::ConstIterator it = properties.begin();
for(; it != properties.end(); ++it) for (; it != properties.end(); ++it)
removeItem(*it); removeItem(*it);
} }
PropertyMap APE::Tag::setProperties(const PropertyMap &origProps) PropertyMap APE::Tag::setProperties(const PropertyMap &origProps) {
{
PropertyMap properties(origProps); // make a local copy that can be modified PropertyMap properties(origProps); // make a local copy that can be modified
// see comment in properties() // see comment in properties()
for(size_t i = 0; i < keyConversionsSize; ++i) for (size_t i = 0; i < keyConversionsSize; ++i)
if(properties.contains(keyConversions[i][0])) { if (properties.contains(keyConversions[i][0])) {
properties.insert(keyConversions[i][1], properties[keyConversions[i][0]]); properties.insert(keyConversions[i][1], properties[keyConversions[i][0]]);
properties.erase(keyConversions[i][0]); properties.erase(keyConversions[i][0]);
} }
@@ -263,31 +345,31 @@ PropertyMap APE::Tag::setProperties(const PropertyMap &origProps)
// first check if tags need to be removed completely // first check if tags need to be removed completely
StringList toRemove; StringList toRemove;
ItemListMap::ConstIterator remIt = itemListMap().begin(); ItemListMap::ConstIterator remIt = itemListMap().begin();
for(; remIt != itemListMap().end(); ++remIt) { for (; remIt != itemListMap().end(); ++remIt) {
String key = remIt->first.upper(); String key = remIt->first.upper();
// only remove if a) key is valid, b) type is text, c) key not contained in new properties // only remove if a) key is valid, b) type is text, c) key not contained in new properties
if(!key.isEmpty() && remIt->second.type() == APE::Item::Text && !properties.contains(key)) if (!key.isEmpty() && remIt->second.type() == APE::Item::Text && !properties.contains(key))
toRemove.append(remIt->first); toRemove.append(remIt->first);
} }
for(StringList::ConstIterator removeIt = toRemove.begin(); removeIt != toRemove.end(); removeIt++) for (StringList::ConstIterator removeIt = toRemove.begin(); removeIt != toRemove.end(); removeIt++)
removeItem(*removeIt); removeItem(*removeIt);
// now sync in the "forward direction" // now sync in the "forward direction"
PropertyMap::ConstIterator it = properties.begin(); PropertyMap::ConstIterator it = properties.begin();
PropertyMap invalid; PropertyMap invalid;
for(; it != properties.end(); ++it) { for (; it != properties.end(); ++it) {
const String &tagName = it->first; const String &tagName = it->first;
if(!checkKey(tagName)) if (!checkKey(tagName))
invalid.insert(it->first, it->second); invalid.insert(it->first, it->second);
else if(!(itemListMap().contains(tagName)) || !(itemListMap()[tagName].values() == it->second)) { else if (!(itemListMap().contains(tagName)) || !(itemListMap()[tagName].values() == it->second)) {
if(it->second.isEmpty()) if (it->second.isEmpty())
removeItem(tagName); removeItem(tagName);
else { else {
StringList::ConstIterator valueIt = it->second.begin(); StringList::ConstIterator valueIt = it->second.begin();
addValue(tagName, *valueIt, true); addValue(tagName, *valueIt, true);
++valueIt; ++valueIt;
for(; valueIt != it->second.end(); ++valueIt) for (; valueIt != it->second.end(); ++valueIt)
addValue(tagName, *valueIt, false); addValue(tagName, *valueIt, false);
} }
} }
@@ -295,35 +377,33 @@ PropertyMap APE::Tag::setProperties(const PropertyMap &origProps)
return invalid; return invalid;
} }
bool APE::Tag::checkKey(const String &key) bool APE::Tag::checkKey(const String &key) {
{
if(key.size() < MinKeyLength || key.size() > MaxKeyLength) if (key.size() < MinKeyLength || key.size() > MaxKeyLength)
return false; return false;
return isKeyValid(key.data(String::UTF8)); return isKeyValid(key.data(String::UTF8));
} }
APE::Footer *APE::Tag::footer() const APE::Footer *APE::Tag::footer() const {
{
return &d->footer; return &d->footer;
} }
const APE::ItemListMap& APE::Tag::itemListMap() const const APE::ItemListMap &APE::Tag::itemListMap() const {
{
return d->itemListMap; return d->itemListMap;
} }
void APE::Tag::removeItem(const String &key) void APE::Tag::removeItem(const String &key) {
{
d->itemListMap.erase(key.upper()); d->itemListMap.erase(key.upper());
} }
void APE::Tag::addValue(const String &key, const String &value, bool replace) void APE::Tag::addValue(const String &key, const String &value, bool replace) {
{
if(replace) if (replace)
removeItem(key); removeItem(key);
if(value.isEmpty()) if (value.isEmpty())
return; return;
// Text items may contain more than one value. // Text items may contain more than one value.
@@ -331,34 +411,36 @@ void APE::Tag::addValue(const String &key, const String &value, bool replace)
ItemListMap::Iterator it = d->itemListMap.find(key.upper()); ItemListMap::Iterator it = d->itemListMap.find(key.upper());
if(it != d->itemListMap.end() && it->second.type() == Item::Text) if (it != d->itemListMap.end() && it->second.type() == Item::Text)
it->second.appendValue(value); it->second.appendValue(value);
else else
setItem(key, Item(key, value)); setItem(key, Item(key, value));
} }
void APE::Tag::setData(const String &key, const ByteVector &value) void APE::Tag::setData(const String &key, const ByteVector &value) {
{
removeItem(key); removeItem(key);
if(value.isEmpty()) if (value.isEmpty())
return; return;
setItem(key, Item(key, value, true)); setItem(key, Item(key, value, true));
} }
void APE::Tag::setItem(const String &key, const Item &item) void APE::Tag::setItem(const String &key, const Item &item) {
{
if(!checkKey(key)) { if (!checkKey(key)) {
debug("APE::Tag::setItem() - Couldn't set an item due to an invalid key."); debug("APE::Tag::setItem() - Couldn't set an item due to an invalid key.");
return; return;
} }
d->itemListMap[key.upper()] = item; d->itemListMap[key.upper()] = item;
} }
bool APE::Tag::isEmpty() const bool APE::Tag::isEmpty() const {
{
return d->itemListMap.isEmpty(); return d->itemListMap.isEmpty();
} }
@@ -366,28 +448,29 @@ bool APE::Tag::isEmpty() const
// protected methods // protected methods
//////////////////////////////////////////////////////////////////////////////// ////////////////////////////////////////////////////////////////////////////////
void APE::Tag::read() void APE::Tag::read() {
{
if(d->file && d->file->isValid()) { if (d->file && d->file->isValid()) {
d->file->seek(d->footerLocation); d->file->seek(d->footerLocation);
d->footer.setData(d->file->readBlock(Footer::size())); d->footer.setData(d->file->readBlock(Footer::size()));
if(d->footer.tagSize() <= Footer::size() || if (d->footer.tagSize() <= Footer::size() ||
d->footer.tagSize() > static_cast<unsigned long>(d->file->length())) d->footer.tagSize() > static_cast<unsigned long>(d->file->length()))
return; return;
d->file->seek(d->footerLocation + Footer::size() - d->footer.tagSize()); d->file->seek(d->footerLocation + Footer::size() - d->footer.tagSize());
parse(d->file->readBlock(d->footer.tagSize() - Footer::size())); parse(d->file->readBlock(d->footer.tagSize() - Footer::size()));
} }
} }
ByteVector APE::Tag::render() const ByteVector APE::Tag::render() const {
{
ByteVector data; ByteVector data;
unsigned int itemCount = 0; unsigned int itemCount = 0;
for(ItemListMap::ConstIterator it = d->itemListMap.begin(); it != d->itemListMap.end(); ++it) { for (ItemListMap::ConstIterator it = d->itemListMap.begin(); it != d->itemListMap.end(); ++it) {
data.append(it->second.render()); data.append(it->second.render());
itemCount++; itemCount++;
} }
@@ -397,32 +480,30 @@ ByteVector APE::Tag::render() const
d->footer.setHeaderPresent(true); d->footer.setHeaderPresent(true);
return d->footer.renderHeader() + data + d->footer.renderFooter(); return d->footer.renderHeader() + data + d->footer.renderFooter();
} }
void APE::Tag::parse(const ByteVector &data) void APE::Tag::parse(const ByteVector &data) {
{
// 11 bytes is the minimum size for an APE item // 11 bytes is the minimum size for an APE item
if(data.size() < 11) if (data.size() < 11)
return; return;
unsigned int pos = 0; size_t pos = 0;
for(unsigned int i = 0; i < d->footer.itemCount() && pos <= data.size() - 11; i++) { for (unsigned int i = 0; i < d->footer.itemCount() && pos <= data.size() - 11; i++) {
const int nullPos = data.find('\0', pos + 8); const size_t nullPos = data.find('\0', pos + 8);
if(nullPos < 0) { if (nullPos == ByteVector::npos()) {
debug("APE::Tag::parse() - Couldn't find a key/value separator. Stopped parsing."); debug("APE::Tag::parse() - Couldn't find a key/value separator. Stopped parsing.");
return; return;
} }
const unsigned int keyLength = nullPos - pos - 8; const size_t keyLength = nullPos - pos - 8;
const unsigned int valLegnth = data.toUInt(pos, false); const size_t valLegnth = data.toUInt32LE(pos);
if(keyLength >= MinKeyLength if (keyLength >= MinKeyLength && keyLength <= MaxKeyLength && isKeyValid(data.mid(pos + 8, keyLength))) {
&& keyLength <= MaxKeyLength
&& isKeyValid(data.mid(pos + 8, keyLength)))
{
APE::Item item; APE::Item item;
item.parse(data.mid(pos)); item.parse(data.mid(pos));
@@ -434,4 +515,5 @@ void APE::Tag::parse(const ByteVector &data)
pos += keyLength + valLegnth + 9; pos += keyLength + valLegnth + 9;
} }
} }

View File

@@ -34,177 +34,183 @@
#include "apeitem.h" #include "apeitem.h"
#define FRONT_COVER "COVER ART (FRONT)"
#define BACK_COVER "COVER ART (BACK)"
namespace Strawberry_TagLib { namespace Strawberry_TagLib {
namespace TagLib { namespace TagLib {
class File; class File;
//! An implementation of the APE tagging format //! An implementation of the APE tagging format
namespace APE { namespace APE {
class Footer; class Footer;
/*! /*!
* A mapping between a list of item names, or keys, and the associated item. * A mapping between a list of item names, or keys, and the associated item.
* *
* \see APE::Tag::itemListMap() * \see APE::Tag::itemListMap()
*/ */
typedef Map<const String, Item> ItemListMap; typedef Map<String, Item> ItemListMap;
//! An APE tag implementation //! An APE tag implementation
class TAGLIB_EXPORT Tag : public Strawberry_TagLib::TagLib::Tag class TAGLIB_EXPORT Tag : public Strawberry_TagLib::TagLib::Tag {
{ public:
public: /*!
/*! * Create an APE tag with default values.
* Create an APE tag with default values. */
*/ explicit Tag();
Tag();
/*! /*!
* Create an APE tag and parse the data in \a file with APE footer at * Create an APE tag and parse the data in \a file with APE footer at
* \a tagOffset. * \a tagOffset.
*/ */
Tag(Strawberry_TagLib::TagLib::File *file, long footerLocation); explicit Tag(Strawberry_TagLib::TagLib::File *file, long long footerLocation);
/*! /*!
* Destroys this Tag instance. * Destroys this Tag instance.
*/ */
virtual ~Tag(); ~Tag() override;
/*! /*!
* Renders the in memory values to a ByteVector suitable for writing to * Renders the in memory values to a ByteVector suitable for writing to the file.
* the file. */
*/ ByteVector render() const;
ByteVector render() const;
/*! /*!
* Returns the string "APETAGEX" suitable for usage in locating the tag in a * Returns the string "APETAGEX" suitable for usage in locating the tag in a file.
* file. */
*/ static ByteVector fileIdentifier();
static ByteVector fileIdentifier();
// Reimplementations. // Reimplementations.
virtual String title() const; String title() const override;
virtual String artist() const; String artist() const override;
virtual String album() const; String album() const override;
virtual String comment() const; String comment() const override;
virtual String genre() const; String genre() const override;
virtual unsigned int year() const; unsigned int year() const override;
virtual unsigned int track() const; unsigned int track() const override;
virtual void setTitle(const String &s); /**
virtual void setArtist(const String &s); * @brief pictures
virtual void setAlbum(const String &s); * According to :
virtual void setComment(const String &s); * http://www.hydrogenaud.io/forums/index.php?showtopic=40603&st=50&p=504669&#entry504669
virtual void setGenre(const String &s); * http://git.videolan.org/?p=vlc.git;a=blob;f=modules/meta_engine/taglib.cpp
virtual void setYear(unsigned int i); * @return
virtual void setTrack(unsigned int i); */
PictureMap pictures() const override;
/*! void setTitle(const String &s) override;
* Implements the unified tag dictionary interface -- export function. void setArtist(const String &s) override;
* APE tags are perfectly compatible with the dictionary interface because they void setAlbum(const String &s) override;
* support both arbitrary tag names and multiple values. Currently only void setComment(const String &s) override;
* APE items of type *Text* are handled by the dictionary interface; all *Binary* void setGenre(const String &s) override;
* and *Locator* items will be put into the unsupportedData list and can be void setYear(unsigned int i) override;
* deleted on request using removeUnsupportedProperties(). The same happens void setTrack(unsigned int i) override;
* to Text items if their key is invalid for PropertyMap (which should actually void setPictures(const PictureMap &l) override;
* never happen).
*
* The only conversion done by this export function is to rename the APE tags
* TRACK to TRACKNUMBER, YEAR to DATE, and ALBUM ARTIST to ALBUMARTIST, respectively,
* in order to be compliant with the names used in other formats.
*/
PropertyMap properties() const;
void removeUnsupportedProperties(const StringList &properties); /*!
* Implements the unified tag dictionary interface -- export function.
* APE tags are perfectly compatible with the dictionary interface because they
* support both arbitrary tag names and multiple values.
* Currently only APE items of type *Text* are handled by the dictionary interface; all *Binary*
* and *Locator* items will be put into the unsupportedData list and can be
* deleted on request using removeUnsupportedProperties().
* The same happens to Text items if their key is invalid for PropertyMap (which should actually never happen).
*
* The only conversion done by this export function is to rename the APE tags
* TRACK to TRACKNUMBER, YEAR to DATE, and ALBUM ARTIST to ALBUMARTIST,
* respectively, in order to be compliant with the names used in other formats.
*/
PropertyMap properties() const override;
/*! void removeUnsupportedProperties(const StringList &properties) override;
* Implements the unified tag dictionary interface -- import function. The same
* comments as for the export function apply; additionally note that the APE tag
* specification requires keys to have between 2 and 16 printable ASCII characters
* with the exception of the fixed strings "ID3", "TAG", "OGGS", and "MP+".
*/
PropertyMap setProperties(const PropertyMap &);
/*! /*!
* Check if the given String is a valid APE tag key. * Implements the unified tag dictionary interface -- import function.
*/ * The same comments as for the export function apply; additionally note that the APE tag
static bool checkKey(const String&); * specification requires keys to have between 2 and 16 printable ASCII characters
* with the exception of the fixed strings "ID3", "TAG", "OGGS", and "MP+".
*/
PropertyMap setProperties(const PropertyMap &) override;
/*! /*!
* Returns a pointer to the tag's footer. * Check if the given String is a valid APE tag key.
*/ */
Footer *footer() const; static bool checkKey(const String &);
/*! /*!
* Returns a reference to the item list map. This is an ItemListMap of * Returns a pointer to the tag's footer.
* all of the items in the tag. */
* Footer *footer() const;
* This is the most powerful structure for accessing the items of the tag.
*
* APE tags are case-insensitive, all keys in this map have been converted
* to upper case.
*
* \warning You should not modify this data structure directly, instead
* use setItem() and removeItem().
*/
const ItemListMap &itemListMap() const;
/*! /*!
* Removes the \a key item from the tag * Returns a reference to the item list map.
*/ * This is an ItemListMap of all of the items in the tag.
void removeItem(const String &key); *
* This is the most powerful structure for accessing the items of the tag.
*
* APE tags are case-insensitive, all keys in this map have been converted
* to upper case.
*
* \warning You should not modify this data structure directly, instead
* use setItem() and removeItem().
*/
const ItemListMap &itemListMap() const;
/*! /*!
* Adds to the text item specified by \a key the data \a value. If \a replace * Removes the \a key item from the tag
* is true, then all of the other values on the same key will be removed */
* first. If a binary item exists for \a key it will be removed first. void removeItem(const String &key);
*/
void addValue(const String &key, const String &value, bool replace = true);
/*! /*!
* Set the binary data for the key specified by \a item to \a value * Adds to the text item specified by \a key the data \a value.
* This will convert the item to type \a Binary if it isn't already and * If \a replace is true, then all of the other values on the same key will be removed first.
* all of the other values on the same key will be removed. * If a binary item exists for \a key it will be removed first.
*/ */
void setData(const String &key, const ByteVector &value); void addValue(const String &key, const String &value, bool replace = true);
/*! /*!
* Sets the \a key item to the value of \a item. If an item with the \a key is already * Set the binary data for the key specified by \a item to \a value
* present, it will be replaced. * This will convert the item to type \a Binary if it isn't already and all of the other values on the same key will be removed.
*/ */
void setItem(const String &key, const Item &item); void setData(const String &key, const ByteVector &value);
/*! /*!
* Returns true if the tag does not contain any data. * Sets the \a key item to the value of \a item. If an item with the \a key is already present, it will be replaced.
*/ */
bool isEmpty() const; void setItem(const String &key, const Item &item);
protected: /*!
* Returns true if the tag does not contain any data.
*/
bool isEmpty() const override;
/*! protected:
* Reads from the file specified in the constructor. /*!
*/ * Reads from the file specified in the constructor.
void read(); */
void read();
/*! /*!
* Parses the body of the tag in \a data. * Parses the body of the tag in \a data.
*/ */
void parse(const ByteVector &data); void parse(const ByteVector &data);
private: private:
Tag(const Tag &); Tag(const Tag &);
Tag &operator=(const Tag &); Tag &operator=(const Tag &);
class TagPrivate; class TagPrivate;
TagPrivate *d; TagPrivate *d;
}; };
} } // namespace APE
} } // namespace TagLib
} } // namespace Strawberry_TagLib
#endif #endif

View File

@@ -23,9 +23,10 @@
* http://www.mozilla.org/MPL/ * * http://www.mozilla.org/MPL/ *
***************************************************************************/ ***************************************************************************/
#include <taglib.h> #include <memory>
#include <tdebug.h>
#include <trefcounter.h> #include "taglib.h"
#include "tdebug.h"
#include "asfattribute.h" #include "asfattribute.h"
#include "asffile.h" #include "asffile.h"
@@ -33,15 +34,11 @@
using namespace Strawberry_TagLib::TagLib; using namespace Strawberry_TagLib::TagLib;
class ASF::Attribute::AttributePrivate : public RefCounter namespace {
{ struct AttributeData {
public: explicit AttributeData() : numericValue(0), stream(0), language(0) {}
AttributePrivate() :
pictureValue(ASF::Picture::fromInvalid()), ASF::Attribute::AttributeTypes type;
numericValue(0),
stream(0),
language(0) {}
AttributeTypes type;
String stringValue; String stringValue;
ByteVector byteVectorValue; ByteVector byteVectorValue;
ASF::Picture pictureValue; ASF::Picture pictureValue;
@@ -49,303 +46,283 @@ public:
int stream; int stream;
int language; int language;
}; };
} // namespace
class ASF::Attribute::AttributePrivate {
public:
AttributePrivate() : data(new AttributeData()) {
data->pictureValue = ASF::Picture::fromInvalid();
}
std::shared_ptr<AttributeData> data;
};
//////////////////////////////////////////////////////////////////////////////// ////////////////////////////////////////////////////////////////////////////////
// public members // public members
//////////////////////////////////////////////////////////////////////////////// ////////////////////////////////////////////////////////////////////////////////
ASF::Attribute::Attribute() : ASF::Attribute::Attribute() : d(new AttributePrivate()) {
d(new AttributePrivate()) d->data->type = UnicodeType;
{
d->type = UnicodeType;
} }
ASF::Attribute::Attribute(const ASF::Attribute &other) : ASF::Attribute::Attribute(const ASF::Attribute &other) : d(new AttributePrivate(*other.d)) {}
d(other.d)
{ ASF::Attribute::Attribute(const String &value) : d(new AttributePrivate()) {
d->ref(); d->data->type = UnicodeType;
d->data->stringValue = value;
} }
ASF::Attribute::Attribute(const String &value) : ASF::Attribute::Attribute(const ByteVector &value) : d(new AttributePrivate()) {
d(new AttributePrivate()) d->data->type = BytesType;
{ d->data->byteVectorValue = value;
d->type = UnicodeType;
d->stringValue = value;
} }
ASF::Attribute::Attribute(const ByteVector &value) : ASF::Attribute::Attribute(const ASF::Picture &value) : d(new AttributePrivate()) {
d(new AttributePrivate()) d->data->type = BytesType;
{ d->data->pictureValue = value;
d->type = BytesType;
d->byteVectorValue = value;
} }
ASF::Attribute::Attribute(const ASF::Picture &value) : ASF::Attribute::Attribute(unsigned int value) : d(new AttributePrivate()) {
d(new AttributePrivate()) d->data->type = DWordType;
{ d->data->numericValue = value;
d->type = BytesType;
d->pictureValue = value;
} }
ASF::Attribute::Attribute(unsigned int value) : ASF::Attribute::Attribute(unsigned long long value) : d(new AttributePrivate()) {
d(new AttributePrivate()) d->data->type = QWordType;
{ d->data->numericValue = value;
d->type = DWordType;
d->numericValue = value;
} }
ASF::Attribute::Attribute(unsigned long long value) : ASF::Attribute::Attribute(unsigned short value) : d(new AttributePrivate()) {
d(new AttributePrivate()) d->data->type = WordType;
{ d->data->numericValue = value;
d->type = QWordType;
d->numericValue = value;
} }
ASF::Attribute::Attribute(unsigned short value) : ASF::Attribute::Attribute(bool value) : d(new AttributePrivate()) {
d(new AttributePrivate()) d->data->type = BoolType;
{ d->data->numericValue = value;
d->type = WordType;
d->numericValue = value;
} }
ASF::Attribute::Attribute(bool value) : ASF::Attribute &ASF::Attribute::operator=(const ASF::Attribute &other) {
d(new AttributePrivate())
{
d->type = BoolType;
d->numericValue = value;
}
ASF::Attribute &ASF::Attribute::operator=(const ASF::Attribute &other)
{
Attribute(other).swap(*this); Attribute(other).swap(*this);
return *this; return *this;
} }
void ASF::Attribute::swap(Attribute &other) void ASF::Attribute::swap(Attribute &other) {
{
using std::swap; using std::swap;
swap(d, other.d); swap(d, other.d);
} }
ASF::Attribute::~Attribute() ASF::Attribute::~Attribute() {
{ delete d;
if(d->deref())
delete d;
} }
ASF::Attribute::AttributeTypes ASF::Attribute::type() const ASF::Attribute::AttributeTypes ASF::Attribute::type() const {
{ return d->data->type;
return d->type;
} }
String ASF::Attribute::toString() const String ASF::Attribute::toString() const {
{ return d->data->stringValue;
return d->stringValue;
} }
ByteVector ASF::Attribute::toByteVector() const ByteVector ASF::Attribute::toByteVector() const {
{ if (d->data->pictureValue.isValid())
if(d->pictureValue.isValid()) return d->data->pictureValue.render();
return d->pictureValue.render();
return d->byteVectorValue; return d->data->byteVectorValue;
} }
unsigned short ASF::Attribute::toBool() const unsigned short ASF::Attribute::toBool() const {
{ return d->data->numericValue ? 1 : 0;
return d->numericValue ? 1 : 0;
} }
unsigned short ASF::Attribute::toUShort() const unsigned short ASF::Attribute::toUShort() const {
{ return static_cast<unsigned short>(d->data->numericValue);
return static_cast<unsigned short>(d->numericValue);
} }
unsigned int ASF::Attribute::toUInt() const unsigned int ASF::Attribute::toUInt() const {
{ return static_cast<unsigned int>(d->data->numericValue);
return static_cast<unsigned int>(d->numericValue);
} }
unsigned long long ASF::Attribute::toULongLong() const unsigned long long ASF::Attribute::toULongLong() const {
{ return static_cast<unsigned long long>(d->data->numericValue);
return static_cast<unsigned long long>(d->numericValue);
} }
ASF::Picture ASF::Attribute::toPicture() const ASF::Picture ASF::Attribute::toPicture() const {
{ return d->data->pictureValue;
return d->pictureValue;
} }
String ASF::Attribute::parse(ASF::File &f, int kind) String ASF::Attribute::parse(ASF::File &f, int kind) {
{
unsigned int size, nameLength; unsigned int size, nameLength;
String name; String name;
d->pictureValue = Picture::fromInvalid(); d->data->pictureValue = Picture::fromInvalid();
// extended content descriptor // extended content descriptor
if(kind == 0) { if (kind == 0) {
nameLength = readWORD(&f); nameLength = readWORD(&f);
name = readString(&f, nameLength); name = readString(&f, nameLength);
d->type = ASF::Attribute::AttributeTypes(readWORD(&f)); d->data->type = ASF::Attribute::AttributeTypes(readWORD(&f));
size = readWORD(&f); size = readWORD(&f);
} }
// metadata & metadata library // metadata & metadata library
else { else {
int temp = readWORD(&f); int temp = readWORD(&f);
// metadata library // metadata library
if(kind == 2) { if (kind == 2) {
d->language = temp; d->data->language = temp;
} }
d->stream = readWORD(&f); d->data->stream = readWORD(&f);
nameLength = readWORD(&f); nameLength = readWORD(&f);
d->type = ASF::Attribute::AttributeTypes(readWORD(&f)); d->data->type = ASF::Attribute::AttributeTypes(readWORD(&f));
size = readDWORD(&f); size = readDWORD(&f);
name = readString(&f, nameLength); name = readString(&f, nameLength);
} }
if(kind != 2 && size > 65535) { if (kind != 2 && size > 65535) {
debug("ASF::Attribute::parse() -- Value larger than 64kB"); debug("ASF::Attribute::parse() -- Value larger than 64kB");
} }
switch(d->type) { switch (d->data->type) {
case WordType: case WordType:
d->numericValue = readWORD(&f); d->data->numericValue = readWORD(&f);
break; break;
case BoolType: case BoolType:
if(kind == 0) { if (kind == 0) {
d->numericValue = (readDWORD(&f) != 0); d->data->numericValue = (readDWORD(&f) != 0);
} }
else { else {
d->numericValue = (readWORD(&f) != 0); d->data->numericValue = (readWORD(&f) != 0);
} }
break; break;
case DWordType: case DWordType:
d->numericValue = readDWORD(&f); d->data->numericValue = readDWORD(&f);
break; break;
case QWordType: case QWordType:
d->numericValue = readQWORD(&f); d->data->numericValue = readQWORD(&f);
break; break;
case UnicodeType: case UnicodeType:
d->stringValue = readString(&f, size); d->data->stringValue = readString(&f, size);
break; break;
case BytesType: case BytesType:
case GuidType: case GuidType:
d->byteVectorValue = f.readBlock(size); d->data->byteVectorValue = f.readBlock(size);
break; break;
} }
if(d->type == BytesType && name == "WM/Picture") { if (d->data->type == BytesType && name == "WM/Picture") {
d->pictureValue.parse(d->byteVectorValue); d->data->pictureValue.parse(d->data->byteVectorValue);
if(d->pictureValue.isValid()) { if (d->data->pictureValue.isValid()) {
d->byteVectorValue.clear(); d->data->byteVectorValue.clear();
} }
} }
return name; return name;
} }
int ASF::Attribute::dataSize() const int ASF::Attribute::dataSize() const {
{
switch (d->type) { switch (d->data->type) {
case WordType: case WordType:
return 2; return 2;
case BoolType: case BoolType:
return 4; return 4;
case DWordType: case DWordType:
return 4; return 4;
case QWordType: case QWordType:
return 5; return 5;
case UnicodeType: case UnicodeType:
return d->stringValue.size() * 2 + 2; return static_cast<int>(d->data->stringValue.size() * 2 + 2);
case BytesType: case BytesType:
if(d->pictureValue.isValid()) if (d->data->pictureValue.isValid())
return d->pictureValue.dataSize(); return d->data->pictureValue.dataSize();
case GuidType: break;
return d->byteVectorValue.size(); case GuidType:
return static_cast<int>(d->data->byteVectorValue.size());
} }
return 0; return 0;
} }
ByteVector ASF::Attribute::render(const String &name, int kind) const ByteVector ASF::Attribute::render(const String &name, int kind) const {
{
ByteVector data; ByteVector data;
switch (d->type) { switch (d->data->type) {
case WordType: case WordType:
data.append(ByteVector::fromShort(toUShort(), false)); data.append(ByteVector::fromUInt16LE(toUShort()));
break; break;
case BoolType: case BoolType:
if(kind == 0) { if (kind == 0) {
data.append(ByteVector::fromUInt(toBool(), false)); data.append(ByteVector::fromUInt32LE(toBool()));
} }
else { else {
data.append(ByteVector::fromShort(toBool(), false)); data.append(ByteVector::fromUInt16LE(toBool()));
} }
break; break;
case DWordType: case DWordType:
data.append(ByteVector::fromUInt(toUInt(), false)); data.append(ByteVector::fromUInt32LE(toUInt()));
break; break;
case QWordType: case QWordType:
data.append(ByteVector::fromLongLong(toULongLong(), false)); data.append(ByteVector::fromUInt64LE(toULongLong()));
break; break;
case UnicodeType: case UnicodeType:
data.append(renderString(d->stringValue)); data.append(renderString(d->data->stringValue));
break; break;
case BytesType: case BytesType:
if(d->pictureValue.isValid()) { if (d->data->pictureValue.isValid()) {
data.append(d->pictureValue.render()); data.append(d->data->pictureValue.render());
break;
}
break;
case GuidType:
data.append(d->data->byteVectorValue);
break; break;
}
case GuidType:
data.append(d->byteVectorValue);
break;
} }
if(kind == 0) { if (kind == 0) {
data = renderString(name, true) + data = renderString(name, true) +
ByteVector::fromShort((int)d->type, false) + ByteVector::fromUInt16LE((int)d->data->type) +
ByteVector::fromShort(data.size(), false) + ByteVector::fromUInt16LE(data.size()) +
data; data;
} }
else { else {
ByteVector nameData = renderString(name); ByteVector nameData = renderString(name);
data = ByteVector::fromShort(kind == 2 ? d->language : 0, false) + data = ByteVector::fromUInt16LE(kind == 2 ? d->data->language : 0) +
ByteVector::fromShort(d->stream, false) + ByteVector::fromUInt16LE(d->data->stream) +
ByteVector::fromShort(nameData.size(), false) + ByteVector::fromUInt16LE(nameData.size()) +
ByteVector::fromShort((int)d->type, false) + ByteVector::fromUInt16LE(static_cast<int>(d->data->type)) +
ByteVector::fromUInt(data.size(), false) + ByteVector::fromUInt32LE(data.size()) +
nameData + nameData +
data; data;
} }
return data; return data;
} }
int ASF::Attribute::language() const int ASF::Attribute::language() const {
{ return d->data->language;
return d->language;
} }
void ASF::Attribute::setLanguage(int value) void ASF::Attribute::setLanguage(int value) {
{ d->data->language = value;
d->language = value;
} }
int ASF::Attribute::stream() const int ASF::Attribute::stream() const {
{ return d->data->stream;
return d->stream;
} }
void ASF::Attribute::setStream(int value) void ASF::Attribute::setStream(int value) {
{ d->data->stream = value;
d->stream = value;
} }

View File

@@ -32,179 +32,169 @@
#include "asfpicture.h" #include "asfpicture.h"
namespace Strawberry_TagLib { namespace Strawberry_TagLib {
namespace TagLib namespace TagLib {
{ namespace ASF {
namespace ASF class File;
{ class Picture;
class File; class TAGLIB_EXPORT Attribute {
class Picture; public:
/*!
* Enum of types an Attribute can have.
*/
enum AttributeTypes {
UnicodeType = 0,
BytesType = 1,
BoolType = 2,
DWordType = 3,
QWordType = 4,
WordType = 5,
GuidType = 6
};
class TAGLIB_EXPORT Attribute /*!
{ * Constructs an empty attribute.
public: */
Attribute();
/*! /*!
* Enum of types an Attribute can have. * Constructs an attribute with \a key and a UnicodeType \a value.
*/ */
enum AttributeTypes { Attribute(const String &value);
UnicodeType = 0,
BytesType = 1,
BoolType = 2,
DWordType = 3,
QWordType = 4,
WordType = 5,
GuidType = 6
};
/*! /*!
* Constructs an empty attribute. * Constructs an attribute with \a key and a BytesType \a value.
*/ */
Attribute(); Attribute(const ByteVector &value);
/*! /*!
* Constructs an attribute with \a key and a UnicodeType \a value. * Constructs an attribute with \a key and a Picture \a value.
*/ *
Attribute(const String &value); * This attribute is compatible with the ID3 frame, APIC. The ID3 specification for the APIC frame stipulates that,
* while there may be any number of APIC frames associated with a file,
* only one may be of type 1 and only one may be of type 2.
*
* The specification also states that the description of the picture can be no longer than 64 characters, but can be empty.
* WM/Picture attributes added with TagLib::ASF are not automatically validated to conform to ID3 specifications.
* You must add code in your application to perform validations if you want to maintain complete compatibility with ID3.
*/
Attribute(const Picture &value);
/*! /*!
* Constructs an attribute with \a key and a BytesType \a value. * Constructs an attribute with \a key and a DWordType \a value.
*/ */
Attribute(const ByteVector &value); Attribute(unsigned int value);
/*! /*!
* Constructs an attribute with \a key and a Picture \a value. * Constructs an attribute with \a key and a QWordType \a value.
* */
* This attribute is compatible with the ID3 frame, APIC. The ID3 specification for the APIC frame stipulates that, Attribute(unsigned long long value);
* while there may be any number of APIC frames associated with a file,
* only one may be of type 1 and only one may be of type 2.
*
* The specification also states that the description of the picture can be no longer than 64 characters, but can be empty.
* WM/Picture attributes added with Strawberry_TagLib::TagLib::ASF are not automatically validated to conform to ID3 specifications.
* You must add code in your application to perform validations if you want to maintain complete compatibility with ID3.
*/
Attribute(const Picture &value);
/*! /*!
* Constructs an attribute with \a key and a DWordType \a value. * Constructs an attribute with \a key and a WordType \a value.
*/ */
Attribute(unsigned int value); Attribute(unsigned short value);
/*! /*!
* Constructs an attribute with \a key and a QWordType \a value. * Constructs an attribute with \a key and a BoolType \a value.
*/ */
Attribute(unsigned long long value); Attribute(bool value);
/*! /*!
* Constructs an attribute with \a key and a WordType \a value. * Construct an attribute as a copy of \a other.
*/ */
Attribute(unsigned short value); Attribute(const Attribute &other);
/*! /*!
* Constructs an attribute with \a key and a BoolType \a value. * Copies the contents of \a other into this item.
*/ */
Attribute(bool value); Attribute &operator=(const Attribute &other);
/*! /*!
* Construct an attribute as a copy of \a other. * Exchanges the content of the Attribute by the content of \a other.
*/ */
Attribute(const Attribute &item); void swap(Attribute &other);
/*! /*!
* Copies the contents of \a other into this item. * Destroys the attribute.
*/ */
Attribute &operator=(const Attribute &other); virtual ~Attribute();
/*! /*!
* Exchanges the content of the Attribute by the content of \a other. * Returns type of the value.
*/ */
void swap(Attribute &other); AttributeTypes type() const;
/*! /*!
* Destroys the attribute. * Returns the BoolType \a value.
*/ */
virtual ~Attribute(); unsigned short toBool() const;
/*! /*!
* Returns type of the value. * Returns the WordType \a value.
*/ */
AttributeTypes type() const; unsigned short toUShort() const;
/*! /*!
* Returns the BoolType \a value. * Returns the DWordType \a value.
*/ */
unsigned short toBool() const; unsigned int toUInt() const;
/*! /*!
* Returns the WordType \a value. * Returns the QWordType \a value.
*/ */
unsigned short toUShort() const; unsigned long long toULongLong() const;
/*! /*!
* Returns the DWordType \a value. * Returns the UnicodeType \a value.
*/ */
unsigned int toUInt() const; String toString() const;
/*! /*!
* Returns the QWordType \a value. * Returns the BytesType \a value.
*/ */
unsigned long long toULongLong() const; ByteVector toByteVector() const;
/*! /*!
* Returns the UnicodeType \a value. * Returns the Picture \a value.
*/ */
String toString() const; Picture toPicture() const;
/*! /*!
* Returns the BytesType \a value. * Returns the language number, or 0 is no stream number was set.
*/ */
ByteVector toByteVector() const; int language() const;
/*! /*!
* Returns the Picture \a value. * Sets the language number.
*/ */
Picture toPicture() const; void setLanguage(int value);
/*! /*!
* Returns the language number, or 0 is no stream number was set. * Returns the stream number, or 0 is no stream number was set.
*/ */
int language() const; int stream() const;
/*! /*!
* Sets the language number. * Sets the stream number.
*/ */
void setLanguage(int value); void setStream(int value);
/*! //! Returns the size of the stored data
* Returns the stream number, or 0 is no stream number was set. int dataSize() const;
*/
int stream() const;
/*! private:
* Sets the stream number. friend class File;
*/
void setStream(int value);
#ifndef DO_NOT_DOCUMENT String parse(ASF::File &file, int kind = 0);
/* THIS IS PRIVATE, DON'T TOUCH IT! */ ByteVector render(const String &name, int kind = 0) const;
String parse(ASF::File &file, int kind = 0);
#endif
//! Returns the size of the stored data class AttributePrivate;
int dataSize() const; AttributePrivate *d;
};
private: } // namespace ASF
friend class File; } // namespace TagLib
} // namespace Strawberry_TagLib
ByteVector render(const String &name, int kind = 0) const;
class AttributePrivate;
AttributePrivate *d;
};
}
}
}
#endif #endif

View File

@@ -23,11 +23,13 @@
* http://www.mozilla.org/MPL/ * * http://www.mozilla.org/MPL/ *
***************************************************************************/ ***************************************************************************/
#include <tdebug.h> #include <memory>
#include <tbytevectorlist.h>
#include <tpropertymap.h> #include "tdebug.h"
#include <tstring.h> #include "tbytevectorlist.h"
#include <tagutils.h> #include "tpropertymap.h"
#include "tstring.h"
#include "tagutils.h"
#include "asffile.h" #include "asffile.h"
#include "asftag.h" #include "asftag.h"
@@ -36,9 +38,8 @@
using namespace Strawberry_TagLib::TagLib; using namespace Strawberry_TagLib::TagLib;
class ASF::File::FilePrivate class ASF::File::FilePrivate {
{ public:
public:
class BaseObject; class BaseObject;
class UnknownObject; class UnknownObject;
class FilePropertiesObject; class FilePropertiesObject;
@@ -50,58 +51,42 @@ public:
class MetadataObject; class MetadataObject;
class MetadataLibraryObject; class MetadataLibraryObject;
FilePrivate(): typedef List<std::shared_ptr<BaseObject>> ObjectList;
headerSize(0), typedef ObjectList::ConstIterator ObjectConstIterator;
tag(0),
properties(0),
contentDescriptionObject(0),
extendedContentDescriptionObject(0),
headerExtensionObject(0),
metadataObject(0),
metadataLibraryObject(0)
{
objects.setAutoDelete(true);
}
~FilePrivate() FilePrivate() : headerSize(0) {}
{
delete tag;
delete properties;
}
unsigned long long headerSize; unsigned long long headerSize;
ASF::Tag *tag; std::unique_ptr<ASF::Tag> tag;
ASF::Properties *properties; std::unique_ptr<ASF::AudioProperties> properties;
List<BaseObject *> objects; ObjectList objects;
ContentDescriptionObject *contentDescriptionObject; std::shared_ptr<ContentDescriptionObject> contentDescriptionObject;
ExtendedContentDescriptionObject *extendedContentDescriptionObject; std::shared_ptr<ExtendedContentDescriptionObject> extendedContentDescriptionObject;
HeaderExtensionObject *headerExtensionObject; std::shared_ptr<HeaderExtensionObject> headerExtensionObject;
MetadataObject *metadataObject; std::shared_ptr<MetadataObject> metadataObject;
MetadataLibraryObject *metadataLibraryObject; std::shared_ptr<MetadataLibraryObject> metadataLibraryObject;
}; };
namespace namespace {
{ const ByteVector headerGuid("\x30\x26\xB2\x75\x8E\x66\xCF\x11\xA6\xD9\x00\xAA\x00\x62\xCE\x6C", 16);
const ByteVector headerGuid("\x30\x26\xB2\x75\x8E\x66\xCF\x11\xA6\xD9\x00\xAA\x00\x62\xCE\x6C", 16); const ByteVector filePropertiesGuid("\xA1\xDC\xAB\x8C\x47\xA9\xCF\x11\x8E\xE4\x00\xC0\x0C\x20\x53\x65", 16);
const ByteVector filePropertiesGuid("\xA1\xDC\xAB\x8C\x47\xA9\xCF\x11\x8E\xE4\x00\xC0\x0C\x20\x53\x65", 16); const ByteVector streamPropertiesGuid("\x91\x07\xDC\xB7\xB7\xA9\xCF\x11\x8E\xE6\x00\xC0\x0C\x20\x53\x65", 16);
const ByteVector streamPropertiesGuid("\x91\x07\xDC\xB7\xB7\xA9\xCF\x11\x8E\xE6\x00\xC0\x0C\x20\x53\x65", 16); const ByteVector contentDescriptionGuid("\x33\x26\xB2\x75\x8E\x66\xCF\x11\xA6\xD9\x00\xAA\x00\x62\xCE\x6C", 16);
const ByteVector contentDescriptionGuid("\x33\x26\xB2\x75\x8E\x66\xCF\x11\xA6\xD9\x00\xAA\x00\x62\xCE\x6C", 16); const ByteVector extendedContentDescriptionGuid("\x40\xA4\xD0\xD2\x07\xE3\xD2\x11\x97\xF0\x00\xA0\xC9\x5E\xA8\x50", 16);
const ByteVector extendedContentDescriptionGuid("\x40\xA4\xD0\xD2\x07\xE3\xD2\x11\x97\xF0\x00\xA0\xC9\x5E\xA8\x50", 16); const ByteVector headerExtensionGuid("\xb5\x03\xbf_.\xa9\xcf\x11\x8e\xe3\x00\xc0\x0c Se", 16);
const ByteVector headerExtensionGuid("\xb5\x03\xbf_.\xa9\xcf\x11\x8e\xe3\x00\xc0\x0c Se", 16); const ByteVector metadataGuid("\xEA\xCB\xF8\xC5\xAF[wH\204g\xAA\214D\xFAL\xCA", 16);
const ByteVector metadataGuid("\xEA\xCB\xF8\xC5\xAF[wH\204g\xAA\214D\xFAL\xCA", 16); const ByteVector metadataLibraryGuid("\224\034#D\230\224\321I\241A\x1d\x13NEpT", 16);
const ByteVector metadataLibraryGuid("\224\034#D\230\224\321I\241A\x1d\x13NEpT", 16); const ByteVector codecListGuid("\x40\x52\xd1\x86\x1d\x31\xd0\x11\xa3\xa4\x00\xa0\xc9\x03\x48\xf6", 16);
const ByteVector codecListGuid("\x40\x52\xd1\x86\x1d\x31\xd0\x11\xa3\xa4\x00\xa0\xc9\x03\x48\xf6", 16); const ByteVector contentEncryptionGuid("\xFB\xB3\x11\x22\x23\xBD\xD2\x11\xB4\xB7\x00\xA0\xC9\x55\xFC\x6E", 16);
const ByteVector contentEncryptionGuid("\xFB\xB3\x11\x22\x23\xBD\xD2\x11\xB4\xB7\x00\xA0\xC9\x55\xFC\x6E", 16); const ByteVector extendedContentEncryptionGuid("\x14\xE6\x8A\x29\x22\x26 \x17\x4C\xB9\x35\xDA\xE0\x7E\xE9\x28\x9C", 16);
const ByteVector extendedContentEncryptionGuid("\x14\xE6\x8A\x29\x22\x26 \x17\x4C\xB9\x35\xDA\xE0\x7E\xE9\x28\x9C", 16); const ByteVector advancedContentEncryptionGuid("\xB6\x9B\x07\x7A\xA4\xDA\x12\x4E\xA5\xCA\x91\xD3\x8D\xC1\x1A\x8D", 16);
const ByteVector advancedContentEncryptionGuid("\xB6\x9B\x07\x7A\xA4\xDA\x12\x4E\xA5\xCA\x91\xD3\x8D\xC1\x1A\x8D", 16); } // namespace
}
class ASF::File::FilePrivate::BaseObject class ASF::File::FilePrivate::BaseObject {
{ public:
public:
ByteVector data; ByteVector data;
virtual ~BaseObject() {} virtual ~BaseObject() {}
virtual ByteVector guid() const = 0; virtual ByteVector guid() const = 0;
@@ -109,355 +94,346 @@ public:
virtual ByteVector render(ASF::File *file); virtual ByteVector render(ASF::File *file);
}; };
class ASF::File::FilePrivate::UnknownObject : public ASF::File::FilePrivate::BaseObject class ASF::File::FilePrivate::UnknownObject : public ASF::File::FilePrivate::BaseObject {
{
ByteVector myGuid; ByteVector myGuid;
public:
public:
explicit UnknownObject(const ByteVector &guid); explicit UnknownObject(const ByteVector &guid);
ByteVector guid() const; ByteVector guid() const override;
}; };
class ASF::File::FilePrivate::FilePropertiesObject : public ASF::File::FilePrivate::BaseObject class ASF::File::FilePrivate::FilePropertiesObject : public ASF::File::FilePrivate::BaseObject {
{ public:
public: ByteVector guid() const override;
ByteVector guid() const; void parse(ASF::File *file, unsigned int size) override;
void parse(ASF::File *file, unsigned int size);
}; };
class ASF::File::FilePrivate::StreamPropertiesObject : public ASF::File::FilePrivate::BaseObject class ASF::File::FilePrivate::StreamPropertiesObject : public ASF::File::FilePrivate::BaseObject {
{ public:
public: ByteVector guid() const override;
ByteVector guid() const; void parse(ASF::File *file, unsigned int size) override;
void parse(ASF::File *file, unsigned int size);
}; };
class ASF::File::FilePrivate::ContentDescriptionObject : public ASF::File::FilePrivate::BaseObject class ASF::File::FilePrivate::ContentDescriptionObject : public ASF::File::FilePrivate::BaseObject {
{ public:
public: ByteVector guid() const override;
ByteVector guid() const; void parse(ASF::File *file, unsigned int size) override;
void parse(ASF::File *file, unsigned int size); ByteVector render(ASF::File *file) override;
ByteVector render(ASF::File *file);
}; };
class ASF::File::FilePrivate::ExtendedContentDescriptionObject : public ASF::File::FilePrivate::BaseObject class ASF::File::FilePrivate::ExtendedContentDescriptionObject : public ASF::File::FilePrivate::BaseObject {
{ public:
public:
ByteVectorList attributeData; ByteVectorList attributeData;
ByteVector guid() const; ByteVector guid() const override;
void parse(ASF::File *file, unsigned int size); void parse(ASF::File *file, unsigned int size) override;
ByteVector render(ASF::File *file); ByteVector render(ASF::File *file) override;
}; };
class ASF::File::FilePrivate::MetadataObject : public ASF::File::FilePrivate::BaseObject class ASF::File::FilePrivate::MetadataObject : public ASF::File::FilePrivate::BaseObject {
{ public:
public:
ByteVectorList attributeData; ByteVectorList attributeData;
ByteVector guid() const; ByteVector guid() const override;
void parse(ASF::File *file, unsigned int size); void parse(ASF::File *file, unsigned int size) override;
ByteVector render(ASF::File *file); ByteVector render(ASF::File *file) override;
}; };
class ASF::File::FilePrivate::MetadataLibraryObject : public ASF::File::FilePrivate::BaseObject class ASF::File::FilePrivate::MetadataLibraryObject : public ASF::File::FilePrivate::BaseObject {
{ public:
public:
ByteVectorList attributeData; ByteVectorList attributeData;
ByteVector guid() const; ByteVector guid() const override;
void parse(ASF::File *file, unsigned int size); void parse(ASF::File *file, unsigned int size) override;
ByteVector render(ASF::File *file); ByteVector render(ASF::File *file) override;
}; };
class ASF::File::FilePrivate::HeaderExtensionObject : public ASF::File::FilePrivate::BaseObject class ASF::File::FilePrivate::HeaderExtensionObject : public ASF::File::FilePrivate::BaseObject {
{ public:
public: ObjectList objects;
List<ASF::File::FilePrivate::BaseObject *> objects;
HeaderExtensionObject(); HeaderExtensionObject();
ByteVector guid() const; ByteVector guid() const override;
void parse(ASF::File *file, unsigned int size); void parse(ASF::File *file, unsigned int size) override;
ByteVector render(ASF::File *file); ByteVector render(ASF::File *file) override;
}; };
class ASF::File::FilePrivate::CodecListObject : public ASF::File::FilePrivate::BaseObject class ASF::File::FilePrivate::CodecListObject : public ASF::File::FilePrivate::BaseObject {
{ public:
public: ByteVector guid() const override;
ByteVector guid() const; void parse(ASF::File *file, unsigned int size) override;
void parse(ASF::File *file, unsigned int size);
private: private:
enum CodecType enum CodecType {
{ Video = 0x0001,
Video = 0x0001, Audio = 0x0002,
Audio = 0x0002,
Unknown = 0xFFFF Unknown = 0xFFFF
}; };
}; };
void ASF::File::FilePrivate::BaseObject::parse(ASF::File *file, unsigned int size) void ASF::File::FilePrivate::BaseObject::parse(ASF::File *file, unsigned int size) {
{
data.clear(); data.clear();
if(size > 24 && size <= (unsigned int)(file->length())) if (size > 24 && static_cast<long long>(size) <= file->length())
data = file->readBlock(size - 24); data = file->readBlock(size - 24);
else else
data = ByteVector(); data = ByteVector();
} }
ByteVector ASF::File::FilePrivate::BaseObject::render(ASF::File * /*file*/) ByteVector ASF::File::FilePrivate::BaseObject::render(ASF::File * /*file*/) {
{ return guid() + ByteVector::fromUInt64LE(data.size() + 24) + data;
return guid() + ByteVector::fromLongLong(data.size() + 24, false) + data;
} }
ASF::File::FilePrivate::UnknownObject::UnknownObject(const ByteVector &guid) : myGuid(guid) ASF::File::FilePrivate::UnknownObject::UnknownObject(const ByteVector &guid) : myGuid(guid) {}
{
}
ByteVector ASF::File::FilePrivate::UnknownObject::guid() const ByteVector ASF::File::FilePrivate::UnknownObject::guid() const {
{
return myGuid; return myGuid;
} }
ByteVector ASF::File::FilePrivate::FilePropertiesObject::guid() const ByteVector ASF::File::FilePrivate::FilePropertiesObject::guid() const {
{
return filePropertiesGuid; return filePropertiesGuid;
} }
void ASF::File::FilePrivate::FilePropertiesObject::parse(ASF::File *file, unsigned int size) void ASF::File::FilePrivate::FilePropertiesObject::parse(ASF::File *file, unsigned int size) {
{
BaseObject::parse(file, size); BaseObject::parse(file, size);
if(data.size() < 64) { if (data.size() < 64) {
debug("ASF::File::FilePrivate::FilePropertiesObject::parse() -- data is too short."); debug("ASF::File::FilePrivate::FilePropertiesObject::parse() -- data is too short.");
return; return;
} }
const long long duration = data.toLongLong(40, false); const long long duration = data.toInt64LE(40);
const long long preroll = data.toLongLong(56, false); const long long preroll = data.toInt64LE(56);
file->d->properties->setLengthInMilliseconds(static_cast<int>(duration / 10000.0 - preroll + 0.5)); file->d->properties->setLengthInMilliseconds(static_cast<int>(duration / 10000.0 - preroll + 0.5));
} }
ByteVector ASF::File::FilePrivate::StreamPropertiesObject::guid() const ByteVector ASF::File::FilePrivate::StreamPropertiesObject::guid() const {
{
return streamPropertiesGuid; return streamPropertiesGuid;
} }
void ASF::File::FilePrivate::StreamPropertiesObject::parse(ASF::File *file, unsigned int size) void ASF::File::FilePrivate::StreamPropertiesObject::parse(ASF::File *file, unsigned int size) {
{
BaseObject::parse(file, size); BaseObject::parse(file, size);
if(data.size() < 70) { if (data.size() < 70) {
debug("ASF::File::FilePrivate::StreamPropertiesObject::parse() -- data is too short."); debug("ASF::File::FilePrivate::StreamPropertiesObject::parse() -- data is too short.");
return; return;
} }
file->d->properties->setCodec(data.toUShort(54, false)); file->d->properties->setCodec(data.toUInt16LE(54));
file->d->properties->setChannels(data.toUShort(56, false)); file->d->properties->setChannels(data.toUInt16LE(56));
file->d->properties->setSampleRate(data.toUInt(58, false)); file->d->properties->setSampleRate(data.toUInt32LE(58));
file->d->properties->setBitrate(static_cast<int>(data.toUInt(62, false) * 8.0 / 1000.0 + 0.5)); file->d->properties->setBitrate(static_cast<int>(data.toUInt32LE(62) * 8.0 / 1000.0 + 0.5));
file->d->properties->setBitsPerSample(data.toUShort(68, false)); file->d->properties->setBitsPerSample(data.toUInt16LE(68));
} }
ByteVector ASF::File::FilePrivate::ContentDescriptionObject::guid() const ByteVector ASF::File::FilePrivate::ContentDescriptionObject::guid() const {
{
return contentDescriptionGuid; return contentDescriptionGuid;
} }
void ASF::File::FilePrivate::ContentDescriptionObject::parse(ASF::File *file, unsigned int /*size*/) void ASF::File::FilePrivate::ContentDescriptionObject::parse(ASF::File *file, unsigned int /*size*/) {
{
const int titleLength = readWORD(file); const int titleLength = readWORD(file);
const int artistLength = readWORD(file); const int artistLength = readWORD(file);
const int copyrightLength = readWORD(file); const int copyrightLength = readWORD(file);
const int commentLength = readWORD(file); const int commentLength = readWORD(file);
const int ratingLength = readWORD(file); const int ratingLength = readWORD(file);
file->d->tag->setTitle(readString(file,titleLength)); file->d->tag->setTitle(readString(file, titleLength));
file->d->tag->setArtist(readString(file,artistLength)); file->d->tag->setArtist(readString(file, artistLength));
file->d->tag->setCopyright(readString(file,copyrightLength)); file->d->tag->setCopyright(readString(file, copyrightLength));
file->d->tag->setComment(readString(file,commentLength)); file->d->tag->setComment(readString(file, commentLength));
file->d->tag->setRating(readString(file,ratingLength)); file->d->tag->setRating(readString(file, ratingLength));
} }
ByteVector ASF::File::FilePrivate::ContentDescriptionObject::render(ASF::File *file) ByteVector ASF::File::FilePrivate::ContentDescriptionObject::render(ASF::File *file) {
{
const ByteVector v1 = renderString(file->d->tag->title()); const ByteVector v1 = renderString(file->d->tag->title());
const ByteVector v2 = renderString(file->d->tag->artist()); const ByteVector v2 = renderString(file->d->tag->artist());
const ByteVector v3 = renderString(file->d->tag->copyright()); const ByteVector v3 = renderString(file->d->tag->copyright());
const ByteVector v4 = renderString(file->d->tag->comment()); const ByteVector v4 = renderString(file->d->tag->comment());
const ByteVector v5 = renderString(file->d->tag->rating()); const ByteVector v5 = renderString(file->d->tag->rating());
data.clear(); data.clear();
data.append(ByteVector::fromShort(v1.size(), false)); data.append(ByteVector::fromUInt16LE(v1.size()));
data.append(ByteVector::fromShort(v2.size(), false)); data.append(ByteVector::fromUInt16LE(v2.size()));
data.append(ByteVector::fromShort(v3.size(), false)); data.append(ByteVector::fromUInt16LE(v3.size()));
data.append(ByteVector::fromShort(v4.size(), false)); data.append(ByteVector::fromUInt16LE(v4.size()));
data.append(ByteVector::fromShort(v5.size(), false)); data.append(ByteVector::fromUInt16LE(v5.size()));
data.append(v1); data.append(v1);
data.append(v2); data.append(v2);
data.append(v3); data.append(v3);
data.append(v4); data.append(v4);
data.append(v5); data.append(v5);
return BaseObject::render(file); return BaseObject::render(file);
} }
ByteVector ASF::File::FilePrivate::ExtendedContentDescriptionObject::guid() const ByteVector ASF::File::FilePrivate::ExtendedContentDescriptionObject::guid() const {
{
return extendedContentDescriptionGuid; return extendedContentDescriptionGuid;
} }
void ASF::File::FilePrivate::ExtendedContentDescriptionObject::parse(ASF::File *file, unsigned int /*size*/) void ASF::File::FilePrivate::ExtendedContentDescriptionObject::parse(ASF::File *file, unsigned int /*size*/) {
{
int count = readWORD(file); int count = readWORD(file);
while(count--) { while (count--) {
ASF::Attribute attribute; ASF::Attribute attribute;
String name = attribute.parse(*file); String name = attribute.parse(*file);
file->d->tag->addAttribute(name, attribute); file->d->tag->addAttribute(name, attribute);
} }
} }
ByteVector ASF::File::FilePrivate::ExtendedContentDescriptionObject::render(ASF::File *file) ByteVector ASF::File::FilePrivate::ExtendedContentDescriptionObject::render(ASF::File *file) {
{
data.clear(); data.clear();
data.append(ByteVector::fromShort(attributeData.size(), false)); data.append(ByteVector::fromUInt16LE(attributeData.size()));
data.append(attributeData.toByteVector("")); data.append(attributeData.toByteVector(""));
return BaseObject::render(file); return BaseObject::render(file);
} }
ByteVector ASF::File::FilePrivate::MetadataObject::guid() const ByteVector ASF::File::FilePrivate::MetadataObject::guid() const {
{
return metadataGuid; return metadataGuid;
} }
void ASF::File::FilePrivate::MetadataObject::parse(ASF::File *file, unsigned int /*size*/) void ASF::File::FilePrivate::MetadataObject::parse(ASF::File *file, unsigned int /*size*/) {
{
int count = readWORD(file); int count = readWORD(file);
while(count--) { while (count--) {
ASF::Attribute attribute; ASF::Attribute attribute;
String name = attribute.parse(*file, 1); String name = attribute.parse(*file, 1);
file->d->tag->addAttribute(name, attribute); file->d->tag->addAttribute(name, attribute);
} }
} }
ByteVector ASF::File::FilePrivate::MetadataObject::render(ASF::File *file) ByteVector ASF::File::FilePrivate::MetadataObject::render(ASF::File *file) {
{
data.clear(); data.clear();
data.append(ByteVector::fromShort(attributeData.size(), false)); data.append(ByteVector::fromUInt16LE(attributeData.size()));
data.append(attributeData.toByteVector("")); data.append(attributeData.toByteVector(""));
return BaseObject::render(file); return BaseObject::render(file);
} }
ByteVector ASF::File::FilePrivate::MetadataLibraryObject::guid() const ByteVector ASF::File::FilePrivate::MetadataLibraryObject::guid() const {
{
return metadataLibraryGuid; return metadataLibraryGuid;
} }
void ASF::File::FilePrivate::MetadataLibraryObject::parse(ASF::File *file, unsigned int /*size*/) void ASF::File::FilePrivate::MetadataLibraryObject::parse(ASF::File *file, unsigned int /*size*/) {
{
int count = readWORD(file); int count = readWORD(file);
while(count--) { while (count--) {
ASF::Attribute attribute; ASF::Attribute attribute;
String name = attribute.parse(*file, 2); String name = attribute.parse(*file, 2);
file->d->tag->addAttribute(name, attribute); file->d->tag->addAttribute(name, attribute);
} }
} }
ByteVector ASF::File::FilePrivate::MetadataLibraryObject::render(ASF::File *file) ByteVector ASF::File::FilePrivate::MetadataLibraryObject::render(ASF::File *file) {
{
data.clear(); data.clear();
data.append(ByteVector::fromShort(attributeData.size(), false)); data.append(ByteVector::fromUInt16LE(attributeData.size()));
data.append(attributeData.toByteVector("")); data.append(attributeData.toByteVector(""));
return BaseObject::render(file); return BaseObject::render(file);
} }
ASF::File::FilePrivate::HeaderExtensionObject::HeaderExtensionObject() ASF::File::FilePrivate::HeaderExtensionObject::HeaderExtensionObject() {
{
objects.setAutoDelete(true); objects.setAutoDelete(true);
} }
ByteVector ASF::File::FilePrivate::HeaderExtensionObject::guid() const ByteVector ASF::File::FilePrivate::HeaderExtensionObject::guid() const {
{
return headerExtensionGuid; return headerExtensionGuid;
} }
void ASF::File::FilePrivate::HeaderExtensionObject::parse(ASF::File *file, unsigned int /*size*/) void ASF::File::FilePrivate::HeaderExtensionObject::parse(ASF::File *file, unsigned int /*size*/) {
{
file->seek(18, File::Current); file->seek(18, File::Current);
long long dataSize = readDWORD(file); long long dataSize = readDWORD(file);
long long dataPos = 0; long long dataPos = 0;
while(dataPos < dataSize) { while (dataPos < dataSize) {
ByteVector guid = file->readBlock(16); ByteVector guid = file->readBlock(16);
if(guid.size() != 16) { if (guid.size() != 16) {
file->setValid(false); file->setValid(false);
break; break;
} }
bool ok; bool ok;
long long size = readQWORD(file, &ok); long long size = readQWORD(file, &ok);
if(!ok) { if (!ok) {
file->setValid(false); file->setValid(false);
break; break;
} }
BaseObject *obj; std::shared_ptr<BaseObject> obj;
if(guid == metadataGuid) { if (guid == metadataGuid) {
file->d->metadataObject = new MetadataObject(); file->d->metadataObject.reset(new MetadataObject());
obj = file->d->metadataObject; obj = file->d->metadataObject;
} }
else if(guid == metadataLibraryGuid) { else if (guid == metadataLibraryGuid) {
file->d->metadataLibraryObject = new MetadataLibraryObject(); file->d->metadataLibraryObject.reset(new MetadataLibraryObject());
obj = file->d->metadataLibraryObject; obj = file->d->metadataLibraryObject;
} }
else { else {
obj = new UnknownObject(guid); obj.reset(new UnknownObject(guid));
} }
obj->parse(file, (unsigned int)size); obj->parse(file, static_cast<unsigned int>(size));
objects.append(obj); objects.append(obj);
dataPos += size; dataPos += size;
} }
} }
ByteVector ASF::File::FilePrivate::HeaderExtensionObject::render(ASF::File *file) ByteVector ASF::File::FilePrivate::HeaderExtensionObject::render(ASF::File *file) {
{
data.clear(); data.clear();
for(List<BaseObject *>::ConstIterator it = objects.begin(); it != objects.end(); ++it) { for (ObjectConstIterator it = objects.begin(); it != objects.end(); ++it) {
data.append((*it)->render(file)); data.append((*it)->render(file));
} }
data = ByteVector("\x11\xD2\xD3\xAB\xBA\xA9\xcf\x11\x8E\xE6\x00\xC0\x0C\x20\x53\x65\x06\x00", 18) + ByteVector::fromUInt(data.size(), false) + data; data = ByteVector("\x11\xD2\xD3\xAB\xBA\xA9\xcf\x11\x8E\xE6\x00\xC0\x0C\x20\x53\x65\x06\x00", 18) + ByteVector::fromUInt32LE(data.size()) + data;
return BaseObject::render(file); return BaseObject::render(file);
} }
ByteVector ASF::File::FilePrivate::CodecListObject::guid() const ByteVector ASF::File::FilePrivate::CodecListObject::guid() const {
{
return codecListGuid; return codecListGuid;
} }
void ASF::File::FilePrivate::CodecListObject::parse(ASF::File *file, unsigned int size) void ASF::File::FilePrivate::CodecListObject::parse(ASF::File *file, unsigned int size) {
{
BaseObject::parse(file, size); BaseObject::parse(file, size);
if(data.size() <= 20) { if (data.size() <= 20) {
debug("ASF::File::FilePrivate::CodecListObject::parse() -- data is too short."); debug("ASF::File::FilePrivate::CodecListObject::parse() -- data is too short.");
return; return;
} }
unsigned int pos = 16; unsigned int pos = 16;
const int count = data.toUInt(pos, false); const int count = data.toUInt32LE(pos);
pos += 4; pos += 4;
for(int i = 0; i < count; ++i) { for (int i = 0; i < count; ++i) {
if(pos >= data.size()) if (pos >= data.size())
break; break;
const CodecType type = static_cast<CodecType>(data.toUShort(pos, false)); const CodecType type = static_cast<CodecType>(data.toUInt16LE(pos));
pos += 2; pos += 2;
int nameLength = data.toUShort(pos, false); int nameLength = data.toUInt16LE(pos);
pos += 2; pos += 2;
const unsigned int namePos = pos; const unsigned int namePos = pos;
pos += nameLength * 2; pos += nameLength * 2;
const int descLength = data.toUShort(pos, false); const int descLength = data.toUInt16LE(pos);
pos += 2; pos += 2;
const unsigned int descPos = pos; const unsigned int descPos = pos;
pos += descLength * 2; pos += descLength * 2;
const int infoLength = data.toUShort(pos, false); const int infoLength = data.toUInt16LE(pos);
pos += 2 + infoLength * 2; pos += 2 + infoLength * 2;
if(type == CodecListObject::Audio) { if (type == CodecListObject::Audio) {
// First audio codec found. // First audio codec found.
const String name(data.mid(namePos, nameLength * 2), String::UTF16LE); const String name(data.mid(namePos, nameLength * 2), String::UTF16LE);
@@ -469,100 +445,78 @@ void ASF::File::FilePrivate::CodecListObject::parse(ASF::File *file, unsigned in
break; break;
} }
} }
} }
//////////////////////////////////////////////////////////////////////////////// ////////////////////////////////////////////////////////////////////////////////
// static members // static members
//////////////////////////////////////////////////////////////////////////////// ////////////////////////////////////////////////////////////////////////////////
bool ASF::File::isSupported(IOStream *stream) bool ASF::File::isSupported(IOStream *stream) {
{
// An ASF file has to start with the designated GUID. // An ASF file has to start with the designated GUID.
const ByteVector id = Utils::readHeader(stream, 16, false); const ByteVector id = Utils::readHeader(stream, 16, false);
return (id == headerGuid); return (id == headerGuid);
} }
//////////////////////////////////////////////////////////////////////////////// ////////////////////////////////////////////////////////////////////////////////
// public members // public members
//////////////////////////////////////////////////////////////////////////////// ////////////////////////////////////////////////////////////////////////////////
ASF::File::File(FileName file, bool, Properties::ReadStyle) : ASF::File::File(FileName file, bool, AudioProperties::ReadStyle) : Strawberry_TagLib::TagLib::File(file), d(new FilePrivate()) {
Strawberry_TagLib::TagLib::File(file), if (isOpen())
d(new FilePrivate())
{
if(isOpen())
read(); read();
} }
ASF::File::File(IOStream *stream, bool, Properties::ReadStyle) : ASF::File::File(IOStream *stream, bool, AudioProperties::ReadStyle) : Strawberry_TagLib::TagLib::File(stream), d(new FilePrivate()) {
Strawberry_TagLib::TagLib::File(stream), if (isOpen())
d(new FilePrivate())
{
if(isOpen())
read(); read();
} }
ASF::File::~File() ASF::File::~File() {
{
delete d; delete d;
} }
ASF::Tag *ASF::File::tag() const ASF::Tag *ASF::File::tag() const {
{ return d->tag.get();
return d->tag;
} }
PropertyMap ASF::File::properties() const ASF::AudioProperties *ASF::File::audioProperties() const {
{ return d->properties.get();
return d->tag->properties();
} }
void ASF::File::removeUnsupportedProperties(const StringList &properties) bool ASF::File::save() {
{
d->tag->removeUnsupportedProperties(properties);
}
PropertyMap ASF::File::setProperties(const PropertyMap &properties) if (readOnly()) {
{
return d->tag->setProperties(properties);
}
ASF::Properties *ASF::File::audioProperties() const
{
return d->properties;
}
bool ASF::File::save()
{
if(readOnly()) {
debug("ASF::File::save() -- File is read only."); debug("ASF::File::save() -- File is read only.");
return false; return false;
} }
if(!isValid()) { if (!isValid()) {
debug("ASF::File::save() -- Trying to save invalid file."); debug("ASF::File::save() -- Trying to save invalid file.");
return false; return false;
} }
if(!d->contentDescriptionObject) { if (!d->contentDescriptionObject) {
d->contentDescriptionObject = new FilePrivate::ContentDescriptionObject(); d->contentDescriptionObject.reset(new FilePrivate::ContentDescriptionObject());
d->objects.append(d->contentDescriptionObject); d->objects.append(d->contentDescriptionObject);
} }
if(!d->extendedContentDescriptionObject) { if (!d->extendedContentDescriptionObject) {
d->extendedContentDescriptionObject = new FilePrivate::ExtendedContentDescriptionObject(); d->extendedContentDescriptionObject.reset(new FilePrivate::ExtendedContentDescriptionObject());
d->objects.append(d->extendedContentDescriptionObject); d->objects.append(d->extendedContentDescriptionObject);
} }
if(!d->headerExtensionObject) { if (!d->headerExtensionObject) {
d->headerExtensionObject = new FilePrivate::HeaderExtensionObject(); d->headerExtensionObject.reset(new FilePrivate::HeaderExtensionObject());
d->objects.append(d->headerExtensionObject); d->objects.append(d->headerExtensionObject);
} }
if(!d->metadataObject) { if (!d->metadataObject) {
d->metadataObject = new FilePrivate::MetadataObject(); d->metadataObject.reset(new FilePrivate::MetadataObject());
d->headerExtensionObject->objects.append(d->metadataObject); d->headerExtensionObject->objects.append(d->metadataObject);
} }
if(!d->metadataLibraryObject) { if (!d->metadataLibraryObject) {
d->metadataLibraryObject = new FilePrivate::MetadataLibraryObject(); d->metadataLibraryObject.reset(new FilePrivate::MetadataLibraryObject());
d->headerExtensionObject->objects.append(d->metadataLibraryObject); d->headerExtensionObject->objects.append(d->metadataLibraryObject);
} }
@@ -572,7 +526,7 @@ bool ASF::File::save()
const AttributeListMap allAttributes = d->tag->attributeListMap(); const AttributeListMap allAttributes = d->tag->attributeListMap();
for(AttributeListMap::ConstIterator it = allAttributes.begin(); it != allAttributes.end(); ++it) { for (AttributeListMap::ConstIterator it = allAttributes.begin(); it != allAttributes.end(); ++it) {
const String &name = it->first; const String &name = it->first;
const AttributeList &attributes = it->second; const AttributeList &attributes = it->second;
@@ -580,17 +534,17 @@ bool ASF::File::save()
bool inExtendedContentDescriptionObject = false; bool inExtendedContentDescriptionObject = false;
bool inMetadataObject = false; bool inMetadataObject = false;
for(AttributeList::ConstIterator jt = attributes.begin(); jt != attributes.end(); ++jt) { for (AttributeList::ConstIterator jt = attributes.begin(); jt != attributes.end(); ++jt) {
const Attribute &attribute = *jt; const Attribute &attribute = *jt;
const bool largeValue = (attribute.dataSize() > 65535); const bool largeValue = (attribute.dataSize() > 65535);
const bool guid = (attribute.type() == Attribute::GuidType); const bool guid = (attribute.type() == Attribute::GuidType);
if(!inExtendedContentDescriptionObject && !guid && !largeValue && attribute.language() == 0 && attribute.stream() == 0) { if (!inExtendedContentDescriptionObject && !guid && !largeValue && attribute.language() == 0 && attribute.stream() == 0) {
d->extendedContentDescriptionObject->attributeData.append(attribute.render(name)); d->extendedContentDescriptionObject->attributeData.append(attribute.render(name));
inExtendedContentDescriptionObject = true; inExtendedContentDescriptionObject = true;
} }
else if(!inMetadataObject && !guid && !largeValue && attribute.language() == 0 && attribute.stream() != 0) { else if (!inMetadataObject && !guid && !largeValue && attribute.language() == 0 && attribute.stream() != 0) {
d->metadataObject->attributeData.append(attribute.render(name, 1)); d->metadataObject->attributeData.append(attribute.render(name, 1));
inMetadataObject = true; inMetadataObject = true;
} }
@@ -601,105 +555,107 @@ bool ASF::File::save()
} }
ByteVector data; ByteVector data;
for(List<FilePrivate::BaseObject *>::ConstIterator it = d->objects.begin(); it != d->objects.end(); ++it) { for (FilePrivate::ObjectConstIterator it = d->objects.begin(); it != d->objects.end(); ++it) {
data.append((*it)->render(this)); data.append((*it)->render(this));
} }
seek(16); seek(16);
writeBlock(ByteVector::fromLongLong(data.size() + 30, false)); writeBlock(ByteVector::fromUInt64LE(data.size() + 30));
writeBlock(ByteVector::fromUInt(d->objects.size(), false)); writeBlock(ByteVector::fromUInt32LE(d->objects.size()));
writeBlock(ByteVector("\x01\x02", 2)); writeBlock(ByteVector("\x01\x02", 2));
insert(data, 30, static_cast<unsigned long>(d->headerSize - 30)); insert(data, 30, static_cast<size_t>(d->headerSize - 30));
d->headerSize = data.size() + 30; d->headerSize = data.size() + 30;
return true; return true;
} }
//////////////////////////////////////////////////////////////////////////////// ////////////////////////////////////////////////////////////////////////////////
// private members // private members
//////////////////////////////////////////////////////////////////////////////// ////////////////////////////////////////////////////////////////////////////////
void ASF::File::read() void ASF::File::read() {
{
if(!isValid()) if (!isValid())
return; return;
if(readBlock(16) != headerGuid) { if (readBlock(16) != headerGuid) {
debug("ASF::File::read(): Not an ASF file."); debug("ASF::File::read(): Not an ASF file.");
setValid(false); setValid(false);
return; return;
} }
d->tag = new ASF::Tag(); d->tag.reset(new ASF::Tag());
d->properties = new ASF::Properties(); d->properties.reset(new ASF::AudioProperties());
bool ok; bool ok;
d->headerSize = readQWORD(this, &ok); d->headerSize = readQWORD(this, &ok);
if(!ok) { if (!ok) {
setValid(false); setValid(false);
return; return;
} }
int numObjects = readDWORD(this, &ok); int numObjects = readDWORD(this, &ok);
if(!ok) { if (!ok) {
setValid(false); setValid(false);
return; return;
} }
seek(2, Current); seek(2, Current);
FilePrivate::FilePropertiesObject *filePropertiesObject = 0; std::shared_ptr<FilePrivate::FilePropertiesObject> filePropertiesObject;
FilePrivate::StreamPropertiesObject *streamPropertiesObject = 0; std::shared_ptr<FilePrivate::StreamPropertiesObject> streamPropertiesObject;
for(int i = 0; i < numObjects; i++) { for (int i = 0; i < numObjects; i++) {
const ByteVector guid = readBlock(16); const ByteVector guid = readBlock(16);
if(guid.size() != 16) { if (guid.size() != 16) {
setValid(false); setValid(false);
break; break;
} }
long size = (long)readQWORD(this, &ok); long size = static_cast<long>(readQWORD(this, &ok));
if(!ok) { if (!ok) {
setValid(false); setValid(false);
break; break;
} }
FilePrivate::BaseObject *obj; std::shared_ptr<FilePrivate::BaseObject> obj;
if(guid == filePropertiesGuid) { if (guid == filePropertiesGuid) {
filePropertiesObject = new FilePrivate::FilePropertiesObject(); filePropertiesObject.reset(new FilePrivate::FilePropertiesObject());
obj = filePropertiesObject; obj = filePropertiesObject;
} }
else if(guid == streamPropertiesGuid) { else if (guid == streamPropertiesGuid) {
streamPropertiesObject = new FilePrivate::StreamPropertiesObject(); streamPropertiesObject.reset(new FilePrivate::StreamPropertiesObject());
obj = streamPropertiesObject; obj = streamPropertiesObject;
} }
else if(guid == contentDescriptionGuid) { else if (guid == contentDescriptionGuid) {
d->contentDescriptionObject = new FilePrivate::ContentDescriptionObject(); d->contentDescriptionObject.reset(new FilePrivate::ContentDescriptionObject());
obj = d->contentDescriptionObject; obj = d->contentDescriptionObject;
} }
else if(guid == extendedContentDescriptionGuid) { else if (guid == extendedContentDescriptionGuid) {
d->extendedContentDescriptionObject = new FilePrivate::ExtendedContentDescriptionObject(); d->extendedContentDescriptionObject.reset(new FilePrivate::ExtendedContentDescriptionObject());
obj = d->extendedContentDescriptionObject; obj = d->extendedContentDescriptionObject;
} }
else if(guid == headerExtensionGuid) { else if (guid == headerExtensionGuid) {
d->headerExtensionObject = new FilePrivate::HeaderExtensionObject(); d->headerExtensionObject.reset(new FilePrivate::HeaderExtensionObject());
obj = d->headerExtensionObject; obj = d->headerExtensionObject;
} }
else if(guid == codecListGuid) { else if (guid == codecListGuid) {
obj = new FilePrivate::CodecListObject(); obj.reset(new FilePrivate::CodecListObject());
} }
else { else {
if(guid == contentEncryptionGuid || if (guid == contentEncryptionGuid ||
guid == extendedContentEncryptionGuid || guid == extendedContentEncryptionGuid ||
guid == advancedContentEncryptionGuid) { guid == advancedContentEncryptionGuid) {
d->properties->setEncrypted(true); d->properties->setEncrypted(true);
} }
obj = new FilePrivate::UnknownObject(guid); obj.reset(new FilePrivate::UnknownObject(guid));
} }
obj->parse(this, size); obj->parse(this, size);
d->objects.append(obj); d->objects.append(obj);
} }
if(!filePropertiesObject || !streamPropertiesObject) { if (!filePropertiesObject || !streamPropertiesObject) {
debug("ASF::File::read(): Missing mandatory header objects."); debug("ASF::File::read(): Missing mandatory header objects.");
setValid(false); setValid(false);
return; return;
} }
} }

View File

@@ -35,106 +35,84 @@
namespace Strawberry_TagLib { namespace Strawberry_TagLib {
namespace TagLib { namespace TagLib {
//! An implementation of ASF (WMA) metadata //! An implementation of ASF (WMA) metadata
namespace ASF { namespace ASF {
/*! /*!
* This implements and provides an interface for ASF files to the * This implements and provides an interface for ASF files to the
* Strawberry_TagLib::TagLib::Tag and Strawberry_TagLib::TagLib::AudioProperties interfaces by way of implementing * TagLib::Tag and TagLib::AudioProperties interfaces by way of implementing
* the abstract Strawberry_TagLib::TagLib::File API as well as providing some additional * the abstract TagLib::File API as well as providing some additional
* information specific to ASF files. * information specific to ASF files.
*/ */
class TAGLIB_EXPORT File : public Strawberry_TagLib::TagLib::File class TAGLIB_EXPORT File : public Strawberry_TagLib::TagLib::File {
{ public:
public: /*!
* Constructs an ASF file from \a file.
*
* \note In the current implementation, both \a readProperties and
* \a propertiesStyle are ignored. The audio properties are always
* read.
*/
explicit File(FileName file, bool readProperties = true, AudioProperties::ReadStyle propertiesStyle = AudioProperties::Average);
/*! /*!
* Constructs an ASF file from \a file. * Constructs an ASF file from \a stream.
* *
* \note In the current implementation, both \a readProperties and * \note In the current implementation, both \a readProperties and
* \a propertiesStyle are ignored. The audio properties are always * \a propertiesStyle are ignored. The audio properties are always
* read. * read.
*/ *
File(FileName file, bool readProperties = true, * \note TagLib will *not* take ownership of the stream, the caller is
Properties::ReadStyle propertiesStyle = Properties::Average); * responsible for deleting it after the File object.
*/
explicit File(IOStream *stream, bool readProperties = true, AudioProperties::ReadStyle propertiesStyle = AudioProperties::Average);
/*! /*!
* Constructs an ASF file from \a stream. * Destroys this instance of the File.
* */
* \note In the current implementation, both \a readProperties and ~File() override;
* \a propertiesStyle are ignored. The audio properties are always
* read.
*
* \note TagLib will *not* take ownership of the stream, the caller is
* responsible for deleting it after the File object.
*/
File(IOStream *stream, bool readProperties = true,
Properties::ReadStyle propertiesStyle = Properties::Average);
/*! /*!
* Destroys this instance of the File. * Returns a pointer to the ASF tag of the file.
*/ *
virtual ~File(); * ASF::Tag implements the tag interface, so this serves as the
* reimplementation of TagLib::File::tag().
*
* \note The Tag <b>is still</b> owned by the ASF::File and should not be
* deleted by the user. It will be deleted when the file (object) is
* destroyed.
*/
Tag *tag() const override;
/*! /*!
* Returns a pointer to the ASF tag of the file. * Returns the ASF audio properties for this file.
* */
* ASF::Tag implements the tag interface, so this serves as the AudioProperties *audioProperties() const override;
* reimplementation of Strawberry_TagLib::TagLib::File::tag().
*
* \note The Tag <b>is still</b> owned by the ASF::File and should not be
* deleted by the user. It will be deleted when the file (object) is
* destroyed.
*/
virtual Tag *tag() const;
/*! /*!
* Implements the unified property interface -- export function. * Save the file.
*/ *
PropertyMap properties() const; * This returns true if the save was successful.
*/
bool save() override;
/*! /*!
* Removes unsupported properties. Forwards to the actual Tag's * Returns whether or not the given \a stream can be opened as an ASF file.
* removeUnsupportedProperties() function. *
*/ * \note This method is designed to do a quick check. The result may not necessarily be correct.
void removeUnsupportedProperties(const StringList &properties); */
static bool isSupported(IOStream *stream);
/*! private:
* Implements the unified property interface -- import function. void read();
*/
PropertyMap setProperties(const PropertyMap &);
/*! class FilePrivate;
* Returns the ASF audio properties for this file. FilePrivate *d;
*/ };
virtual Properties *audioProperties() const;
/*! } // namespace ASF
* Save the file.
*
* This returns true if the save was successful.
*/
virtual bool save();
/*! } // namespace TagLib
* Returns whether or not the given \a stream can be opened as an ASF } // namespace Strawberry_TagLib
* file.
*
* \note This method is designed to do a quick check. The result may
* not necessarily be correct.
*/
static bool isSupported(IOStream *stream);
private:
void read();
class FilePrivate;
FilePrivate *d;
};
}
}
}
#endif #endif

View File

@@ -23,9 +23,10 @@
* http://www.mozilla.org/MPL/ * * http://www.mozilla.org/MPL/ *
***************************************************************************/ ***************************************************************************/
#include <taglib.h> #include <memory>
#include <tdebug.h>
#include <trefcounter.h> #include "taglib.h"
#include "tdebug.h"
#include "asfattribute.h" #include "asfattribute.h"
#include "asffile.h" #include "asffile.h"
@@ -34,150 +35,134 @@
using namespace Strawberry_TagLib::TagLib; using namespace Strawberry_TagLib::TagLib;
class ASF::Picture::PicturePrivate : public RefCounter namespace {
{ struct PictureData {
public:
bool valid; bool valid;
Type type; ASF::Picture::Type type;
String mimeType; String mimeType;
String description; String description;
ByteVector picture; ByteVector picture;
}; };
} // namespace
class ASF::Picture::PicturePrivate {
public:
explicit PicturePrivate() : data(new PictureData()) {}
std::shared_ptr<PictureData> data;
};
//////////////////////////////////////////////////////////////////////////////// ////////////////////////////////////////////////////////////////////////////////
// Picture class members // Picture class members
//////////////////////////////////////////////////////////////////////////////// ////////////////////////////////////////////////////////////////////////////////
ASF::Picture::Picture() : ASF::Picture::Picture() : d(new PicturePrivate()) {
d(new PicturePrivate()) d->data->valid = true;
{
d->valid = true;
} }
ASF::Picture::Picture(const Picture& other) : ASF::Picture::Picture(const Picture& other) : d(new PicturePrivate(*other.d)) {}
d(other.d)
{ ASF::Picture::~Picture() {
d->ref(); delete d;
} }
ASF::Picture::~Picture() bool ASF::Picture::isValid() const {
{ return d->data->valid;
if(d->deref())
delete d;
} }
bool ASF::Picture::isValid() const String ASF::Picture::mimeType() const {
{ return d->data->mimeType;
return d->valid;
} }
String ASF::Picture::mimeType() const void ASF::Picture::setMimeType(const String& value) {
{ d->data->mimeType = value;
return d->mimeType;
} }
void ASF::Picture::setMimeType(const String &value) ASF::Picture::Type ASF::Picture::type() const {
{ return d->data->type;
d->mimeType = value;
} }
ASF::Picture::Type ASF::Picture::type() const void ASF::Picture::setType(const ASF::Picture::Type &t) {
{ d->data->type = t;
return d->type;
} }
void ASF::Picture::setType(const ASF::Picture::Type& t) String ASF::Picture::description() const {
{ return d->data->description;
d->type = t;
} }
String ASF::Picture::description() const void ASF::Picture::setDescription(const String& desc) {
{ d->data->description = desc;
return d->description;
} }
void ASF::Picture::setDescription(const String &desc) ByteVector ASF::Picture::picture() const {
{ return d->data->picture;
d->description = desc;
} }
ByteVector ASF::Picture::picture() const void ASF::Picture::setPicture(const ByteVector &p) {
{ d->data->picture = p;
return d->picture;
} }
void ASF::Picture::setPicture(const ByteVector &p) int ASF::Picture::dataSize() const {
{ return 9 + (d->data->mimeType.length() + d->data->description.length()) * 2 + d->data->picture.size();
d->picture = p;
} }
int ASF::Picture::dataSize() const ASF::Picture& ASF::Picture::operator=(const ASF::Picture& other) {
{
return
9 + (d->mimeType.length() + d->description.length()) * 2 +
d->picture.size();
}
ASF::Picture& ASF::Picture::operator=(const ASF::Picture& other)
{
Picture(other).swap(*this); Picture(other).swap(*this);
return *this; return *this;
} }
void ASF::Picture::swap(Picture &other) void ASF::Picture::swap(Picture& other) {
{
using std::swap; using std::swap;
swap(d, other.d); swap(d, other.d);
} }
ByteVector ASF::Picture::render() const ByteVector ASF::Picture::render() const {
{
if(!isValid()) if (!isValid())
return ByteVector(); return ByteVector();
return return ByteVector(static_cast<char>(d->data->type)) + ByteVector::fromUInt32LE(d->data->picture.size()) + renderString(d->data->mimeType) + renderString(d->data->description) + d->data->picture;
ByteVector((char)d->type) +
ByteVector::fromUInt(d->picture.size(), false) +
renderString(d->mimeType) +
renderString(d->description) +
d->picture;
} }
void ASF::Picture::parse(const ByteVector& bytes) void ASF::Picture::parse(const ByteVector &bytes) {
{
d->valid = false; d->data->valid = false;
if(bytes.size() < 9) if (bytes.size() < 9)
return; return;
int pos = 0; size_t pos = 0;
d->type = (Type)bytes[0]; ++pos; d->data->type = static_cast<Type>(bytes[0]);
const unsigned int dataLen = bytes.toUInt(pos, false); pos+=4; ++pos;
const unsigned int dataLen = bytes.toUInt32LE(pos);
pos += 4;
const ByteVector nullStringTerminator(2, 0); const ByteVector nullStringTerminator(2, 0);
int endPos = bytes.find(nullStringTerminator, pos, 2); size_t endPos = bytes.find(nullStringTerminator, pos, 2);
if(endPos < 0) if (endPos == ByteVector::npos())
return; return;
d->mimeType = String(bytes.mid(pos, endPos - pos), String::UTF16LE); d->data->mimeType = String(bytes.mid(pos, endPos - pos), String::UTF16LE);
pos = endPos+2; pos = endPos + 2;
endPos = bytes.find(nullStringTerminator, pos, 2); endPos = bytes.find(nullStringTerminator, pos, 2);
if(endPos < 0) if (endPos == ByteVector::npos())
return; return;
d->description = String(bytes.mid(pos, endPos - pos), String::UTF16LE); d->data->description = String(bytes.mid(pos, endPos - pos), String::UTF16LE);
pos = endPos+2; pos = endPos + 2;
if(dataLen + pos != bytes.size()) if (dataLen + pos != bytes.size())
return; return;
d->picture = bytes.mid(pos, dataLen); d->data->picture = bytes.mid(pos, dataLen);
d->valid = true; d->data->valid = true;
return;
} }
ASF::Picture ASF::Picture::fromInvalid() ASF::Picture ASF::Picture::fromInvalid() {
{
Picture ret; Picture ret;
ret.d->valid = false; ret.d->data->valid = false;
return ret; return ret;
} }

View File

@@ -32,193 +32,187 @@
#include "attachedpictureframe.h" #include "attachedpictureframe.h"
namespace Strawberry_TagLib { namespace Strawberry_TagLib {
namespace TagLib namespace TagLib {
{ namespace ASF {
namespace ASF class Attribute;
{
//! An ASF attached picture interface implementation //! An ASF attached picture interface implementation
/*! /*!
* This is an implementation of ASF attached pictures interface. Pictures may be * This is an implementation of ASF attached pictures interface.
* included in attributes, one per WM/Picture attribute (but there may be multiple WM/Picture * Pictures may be included in attributes, one per WM/Picture attribute (but there may be multiple WM/Picture attribute in a single tag).
* attribute in a single tag). These pictures are usually in either JPEG or * These pictures are usually in either JPEG or PNG format.
* PNG format. * \see Attribute::toPicture()
* \see Attribute::toPicture() * \see Attribute::Attribute(const Picture& picture)
* \see Attribute::Attribute(const Picture& picture) */
*/ class TAGLIB_EXPORT Picture {
class TAGLIB_EXPORT Picture { public:
public: /*!
* This describes the function or content of the picture.
*/
enum Type {
//! A type not enumerated below
Other = 0x00,
//! 32x32 PNG image that should be used as the file icon
FileIcon = 0x01,
//! File icon of a different size or format
OtherFileIcon = 0x02,
//! Front cover image of the album
FrontCover = 0x03,
//! Back cover image of the album
BackCover = 0x04,
//! Inside leaflet page of the album
LeafletPage = 0x05,
//! Image from the album itself
Media = 0x06,
//! Picture of the lead artist or soloist
LeadArtist = 0x07,
//! Picture of the artist or performer
Artist = 0x08,
//! Picture of the conductor
Conductor = 0x09,
//! Picture of the band or orchestra
Band = 0x0A,
//! Picture of the composer
Composer = 0x0B,
//! Picture of the lyricist or text writer
Lyricist = 0x0C,
//! Picture of the recording location or studio
RecordingLocation = 0x0D,
//! Picture of the artists during recording
DuringRecording = 0x0E,
//! Picture of the artists during performance
DuringPerformance = 0x0F,
//! Picture from a movie or video related to the track
MovieScreenCapture = 0x10,
//! Picture of a large, coloured fish
ColouredFish = 0x11,
//! Illustration related to the track
Illustration = 0x12,
//! Logo of the band or performer
BandLogo = 0x13,
//! Logo of the publisher (record company)
PublisherLogo = 0x14
};
/*! /*!
* This describes the function or content of the picture. * Constructs an empty picture.
*/ */
enum Type { explicit Picture();
//! A type not enumerated below
Other = 0x00,
//! 32x32 PNG image that should be used as the file icon
FileIcon = 0x01,
//! File icon of a different size or format
OtherFileIcon = 0x02,
//! Front cover image of the album
FrontCover = 0x03,
//! Back cover image of the album
BackCover = 0x04,
//! Inside leaflet page of the album
LeafletPage = 0x05,
//! Image from the album itself
Media = 0x06,
//! Picture of the lead artist or soloist
LeadArtist = 0x07,
//! Picture of the artist or performer
Artist = 0x08,
//! Picture of the conductor
Conductor = 0x09,
//! Picture of the band or orchestra
Band = 0x0A,
//! Picture of the composer
Composer = 0x0B,
//! Picture of the lyricist or text writer
Lyricist = 0x0C,
//! Picture of the recording location or studio
RecordingLocation = 0x0D,
//! Picture of the artists during recording
DuringRecording = 0x0E,
//! Picture of the artists during performance
DuringPerformance = 0x0F,
//! Picture from a movie or video related to the track
MovieScreenCapture = 0x10,
//! Picture of a large, coloured fish
ColouredFish = 0x11,
//! Illustration related to the track
Illustration = 0x12,
//! Logo of the band or performer
BandLogo = 0x13,
//! Logo of the publisher (record company)
PublisherLogo = 0x14
};
/*! /*!
* Constructs an empty picture. * Construct an picture as a copy of \a other.
*/ */
Picture(); Picture(const Picture& other);
/*! /*!
* Construct an picture as a copy of \a other. * Destroys the picture.
*/ */
Picture(const Picture& other); virtual ~Picture();
/*! /*!
* Destroys the picture. * Copies the contents of \a other into this picture.
*/ */
virtual ~Picture(); Picture& operator=(const Picture& other);
/*! /*!
* Copies the contents of \a other into this picture. * Exchanges the content of the Picture by the content of \a other.
*/ */
Picture& operator=(const Picture& other); void swap(Picture& other);
/*! /*!
* Exchanges the content of the Picture by the content of \a other. * Returns true if Picture stores valid picture
*/ */
void swap(Picture &other); bool isValid() const;
/*! /*!
* Returns true if Picture stores valid picture * Returns the mime type of the image. This should in most cases be "image/png" or "image/jpeg".
*/ * \see setMimeType(const String &)
bool isValid() const; * \see picture()
* \see setPicture(const ByteArray&)
*/
String mimeType() const;
/*! /*!
* Returns the mime type of the image. This should in most cases be * Sets the mime type of the image. This should in most cases be "image/png" or "image/jpeg".
* "image/png" or "image/jpeg". * \see setMimeType(const String &)
* \see setMimeType(const String &) * \see picture()
* \see picture() * \see setPicture(const ByteArray&)
* \see setPicture(const ByteArray&) */
*/ void setMimeType(const String& value);
String mimeType() const;
/*! /*!
* Sets the mime type of the image. This should in most cases be * Returns the type of the image.
* "image/png" or "image/jpeg". *
* \see setMimeType(const String &) * \see Type
* \see picture() * \see setType()
* \see setPicture(const ByteArray&) */
*/ Type type() const;
void setMimeType(const String &value);
/*! /*!
* Returns the type of the image. * Sets the type for the image.
* *
* \see Type * \see Type
* \see setType() * \see type()
*/ */
Type type() const; void setType(const ASF::Picture::Type& t);
/*! /*!
* Sets the type for the image. * Returns a text description of the image.
* *
* \see Type * \see setDescription()
* \see type() */
*/ String description() const;
void setType(const ASF::Picture::Type& t);
/*! /*!
* Returns a text description of the image. * Sets a textual description of the image to \a desc.
* *
* \see setDescription() * \see description()
*/ */
String description() const; void setDescription(const String& desc);
/*! /*!
* Sets a textual description of the image to \a desc. * Returns the image data as a ByteVector.
* *
* \see description() * \note ByteVector has a data() method that returns a const char * which should make it easy to export this data to external programs.
*/ *
void setDescription(const String &desc); * \see setPicture()
* \see mimeType()
*/
ByteVector picture() const;
/*! /*!
* Returns the image data as a ByteVector. * Sets the image data to \a p.
* * \a p should be of the type specified in this frame's mime-type specification.
* \note ByteVector has a data() method that returns a const char * which *
* should make it easy to export this data to external programs. * \see picture()
* * \see mimeType()
* \see setPicture() * \see setMimeType()
* \see mimeType() */
*/ void setPicture(const ByteVector& p);
ByteVector picture() const;
/*! /*!
* Sets the image data to \a p. \a p should be of the type specified in * Returns picture as binary raw data \a value
* this frame's mime-type specification. */
* ByteVector render() const;
* \see picture()
* \see mimeType()
* \see setMimeType()
*/
void setPicture(const ByteVector &p);
/*! /*!
* Returns picture as binary raw data \a value * Returns picture as binary raw data \a value
*/ */
ByteVector render() const; int dataSize() const;
/*! private:
* Returns picture as binary raw data \a value friend class Attribute;
*/
int dataSize() const;
#ifndef DO_NOT_DOCUMENT void parse(const ByteVector&);
/* THIS IS PRIVATE, DON'T TOUCH IT! */ static Picture fromInvalid();
void parse(const ByteVector& );
static Picture fromInvalid();
#endif
private: private:
class PicturePrivate; class PicturePrivate;
PicturePrivate *d; PicturePrivate* d;
}; };
} } // namespace ASF
} } // namespace TagLib
} } // namespace Strawberry_TagLib
#endif // ASFPICTURE_H #endif // ASFPICTURE_H

View File

@@ -23,30 +23,28 @@
* http://www.mozilla.org/MPL/ * * http://www.mozilla.org/MPL/ *
***************************************************************************/ ***************************************************************************/
#include <tdebug.h> #include "tdebug.h"
#include <tstring.h> #include "tstring.h"
#include "asfproperties.h" #include "asfproperties.h"
using namespace Strawberry_TagLib::TagLib; using namespace Strawberry_TagLib::TagLib;
class ASF::Properties::PropertiesPrivate class ASF::AudioProperties::AudioPropertiesPrivate {
{ public:
public: explicit AudioPropertiesPrivate() : length(0),
PropertiesPrivate() : bitrate(0),
length(0), sampleRate(0),
bitrate(0), channels(0),
sampleRate(0), bitsPerSample(0),
channels(0), codec(ASF::AudioProperties::Unknown),
bitsPerSample(0), encrypted(false) {}
codec(ASF::Properties::Unknown),
encrypted(false) {}
int length; int length;
int bitrate; int bitrate;
int sampleRate; int sampleRate;
int channels; int channels;
int bitsPerSample; int bitsPerSample;
ASF::Properties::Codec codec; ASF::AudioProperties::Codec codec;
String codecName; String codecName;
String codecDescription; String codecDescription;
bool encrypted; bool encrypted;
@@ -56,69 +54,50 @@ public:
// public members // public members
//////////////////////////////////////////////////////////////////////////////// ////////////////////////////////////////////////////////////////////////////////
ASF::Properties::Properties() : ASF::AudioProperties::AudioProperties() : Strawberry_TagLib::TagLib::AudioProperties(), d(new AudioPropertiesPrivate()) {
AudioProperties(AudioProperties::Average),
d(new PropertiesPrivate())
{
} }
ASF::Properties::~Properties() ASF::AudioProperties::~AudioProperties() {
{
delete d; delete d;
} }
int ASF::Properties::length() const int ASF::AudioProperties::lengthInSeconds() const {
{
return lengthInSeconds();
}
int ASF::Properties::lengthInSeconds() const
{
return d->length / 1000; return d->length / 1000;
} }
int ASF::Properties::lengthInMilliseconds() const int ASF::AudioProperties::lengthInMilliseconds() const {
{
return d->length; return d->length;
} }
int ASF::Properties::bitrate() const int ASF::AudioProperties::bitrate() const {
{
return d->bitrate; return d->bitrate;
} }
int ASF::Properties::sampleRate() const int ASF::AudioProperties::sampleRate() const {
{
return d->sampleRate; return d->sampleRate;
} }
int ASF::Properties::channels() const int ASF::AudioProperties::channels() const {
{
return d->channels; return d->channels;
} }
int ASF::Properties::bitsPerSample() const int ASF::AudioProperties::bitsPerSample() const {
{
return d->bitsPerSample; return d->bitsPerSample;
} }
ASF::Properties::Codec ASF::Properties::codec() const ASF::AudioProperties::Codec ASF::AudioProperties::codec() const {
{
return d->codec; return d->codec;
} }
String ASF::Properties::codecName() const String ASF::AudioProperties::codecName() const {
{
return d->codecName; return d->codecName;
} }
String ASF::Properties::codecDescription() const String ASF::AudioProperties::codecDescription() const {
{
return d->codecDescription; return d->codecDescription;
} }
bool ASF::Properties::isEncrypted() const bool ASF::AudioProperties::isEncrypted() const {
{
return d->encrypted; return d->encrypted;
} }
@@ -126,69 +105,56 @@ bool ASF::Properties::isEncrypted() const
// private members // private members
//////////////////////////////////////////////////////////////////////////////// ////////////////////////////////////////////////////////////////////////////////
void ASF::Properties::setLength(int /*length*/) void ASF::AudioProperties::setLengthInMilliseconds(int value) {
{
debug("ASF::Properties::setLength() -- This method is deprecated. Do not use.");
}
void ASF::Properties::setLengthInMilliseconds(int value)
{
d->length = value; d->length = value;
} }
void ASF::Properties::setBitrate(int value) void ASF::AudioProperties::setBitrate(int value) {
{
d->bitrate = value; d->bitrate = value;
} }
void ASF::Properties::setSampleRate(int value) void ASF::AudioProperties::setSampleRate(int value) {
{
d->sampleRate = value; d->sampleRate = value;
} }
void ASF::Properties::setChannels(int value) void ASF::AudioProperties::setChannels(int value) {
{
d->channels = value; d->channels = value;
} }
void ASF::Properties::setBitsPerSample(int value) void ASF::AudioProperties::setBitsPerSample(int value) {
{
d->bitsPerSample = value; d->bitsPerSample = value;
} }
void ASF::Properties::setCodec(int value) void ASF::AudioProperties::setCodec(int value) {
{
switch(value) switch (value) {
{ case 0x0160:
case 0x0160: d->codec = WMA1;
d->codec = WMA1; break;
break; case 0x0161:
case 0x0161: d->codec = WMA2;
d->codec = WMA2; break;
break; case 0x0162:
case 0x0162: d->codec = WMA9Pro;
d->codec = WMA9Pro; break;
break; case 0x0163:
case 0x0163: d->codec = WMA9Lossless;
d->codec = WMA9Lossless; break;
break; default:
default: d->codec = Unknown;
d->codec = Unknown; break;
break;
} }
} }
void ASF::Properties::setCodecName(const String &value) void ASF::AudioProperties::setCodecName(const String &value) {
{
d->codecName = value; d->codecName = value;
} }
void ASF::Properties::setCodecDescription(const String &value) void ASF::AudioProperties::setCodecDescription(const String &value) {
{
d->codecDescription = value; d->codecDescription = value;
} }
void ASF::Properties::setEncrypted(bool value) void ASF::AudioProperties::setEncrypted(bool value) {
{
d->encrypted = value; d->encrypted = value;
} }

View File

@@ -32,157 +32,137 @@
namespace Strawberry_TagLib { namespace Strawberry_TagLib {
namespace TagLib { namespace TagLib {
namespace ASF {
namespace ASF { //! An implementation of ASF audio properties
class TAGLIB_EXPORT AudioProperties : public Strawberry_TagLib::TagLib::AudioProperties {
friend class File;
//! An implementation of ASF audio properties public:
class TAGLIB_EXPORT Properties : public AudioProperties /*!
{ * Audio codec types can be used in ASF file.
public: */
enum Codec {
/*!
* Couldn't detect the codec.
*/
Unknown = 0,
/*! /*!
* Audio codec types can be used in ASF file. * Windows Media Audio 1
*/ */
enum Codec WMA1,
{
/*!
* Couldn't detect the codec.
*/
Unknown = 0,
/*! /*!
* Windows Media Audio 1 * Windows Media Audio 2 or above
*/ */
WMA1, WMA2,
/*! /*!
* Windows Media Audio 2 or above * Windows Media Audio 9 Professional
*/ */
WMA2, WMA9Pro,
/*! /*!
* Windows Media Audio 9 Professional * Windows Media Audio 9 Lossless
*/ */
WMA9Pro, WMA9Lossless,
};
/*! /*!
* Windows Media Audio 9 Lossless * Creates an instance of ASF::Properties.
*/ */
WMA9Lossless, explicit AudioProperties();
};
/*! /*!
* Creates an instance of ASF::Properties. * Destroys this ASF::AudioProperties instance.
*/ */
Properties(); ~AudioProperties() override;
/*! /*!
* Destroys this ASF::Properties instance. * Returns the length of the file in seconds. The length is rounded down to
*/ * the nearest whole second.
virtual ~Properties(); *
* \see lengthInMilliseconds()
*/
int lengthInSeconds() const override;
/*! /*!
* Returns the length of the file in seconds. The length is rounded down to * Returns the length of the file in milliseconds.
* the nearest whole second. *
* * \see lengthInSeconds()
* \note This method is just an alias of lengthInSeconds(). */
* int lengthInMilliseconds() const override;
* \deprecated
*/
virtual int length() const;
/*! /*!
* Returns the length of the file in seconds. The length is rounded down to * Returns the average bit rate of the file in kb/s.
* the nearest whole second. */
* int bitrate() const override;
* \see lengthInMilliseconds()
*/
// BIC: make virtual
int lengthInSeconds() const;
/*! /*!
* Returns the length of the file in milliseconds. * Returns the sample rate in Hz.
* */
* \see lengthInSeconds() int sampleRate() const override;
*/
// BIC: make virtual
int lengthInMilliseconds() const;
/*! /*!
* Returns the average bit rate of the file in kb/s. * Returns the number of audio channels.
*/ */
virtual int bitrate() const; int channels() const override;
/*! /*!
* Returns the sample rate in Hz. * Returns the number of bits per audio sample.
*/ */
virtual int sampleRate() const; int bitsPerSample() const;
/*! /*!
* Returns the number of audio channels. * Returns the codec used in the file.
*/ *
virtual int channels() const; * \see codecName()
* \see codecDescription()
*/
Codec codec() const;
/*! /*!
* Returns the number of bits per audio sample. * Returns the concrete codec name, for example "Windows Media Audio 9.1" used in the file if available, otherwise an empty string.
*/ *
int bitsPerSample() const; * \see codec()
* \see codecDescription()
*/
String codecName() const;
/*! /*!
* Returns the codec used in the file. * Returns the codec description, typically contains the encoder settings,
* * for example "VBR Quality 50, 44kHz, stereo 1-pass VBR" if available, otherwise an empty string.
* \see codecName() *
* \see codecDescription() * \see codec()
*/ * \see codecName()
Codec codec() const; */
String codecDescription() const;
/*! /*!
* Returns the concrete codec name, for example "Windows Media Audio 9.1" * Returns whether or not the file is encrypted.
* used in the file if available, otherwise an empty string. */
* bool isEncrypted() const;
* \see codec()
* \see codecDescription()
*/
String codecName() const;
/*! private:
* Returns the codec description, typically contains the encoder settings, void setLengthInMilliseconds(int value);
* for example "VBR Quality 50, 44kHz, stereo 1-pass VBR" if available, void setBitrate(int value);
* otherwise an empty string. void setSampleRate(int value);
* void setChannels(int value);
* \see codec() void setBitsPerSample(int value);
* \see codecName() void setCodec(int value);
*/ void setCodecName(const String &value);
String codecDescription() const; void setCodecDescription(const String &value);
void setEncrypted(bool value);
/*! private:
* Returns whether or not the file is encrypted. class AudioPropertiesPrivate;
*/ AudioPropertiesPrivate *d;
bool isEncrypted() const; };
#ifndef DO_NOT_DOCUMENT } // namespace ASF
// deprecated
void setLength(int value);
void setLengthInMilliseconds(int value); } // namespace TagLib
void setBitrate(int value); } // namespace Strawberry_TagLib
void setSampleRate(int value);
void setChannels(int value);
void setBitsPerSample(int value);
void setCodec(int value);
void setCodecName(const String &value);
void setCodecDescription(const String &value);
void setEncrypted(bool value);
#endif
private:
class PropertiesPrivate;
PropertiesPrivate *d;
};
}
}
}
#endif #endif

View File

@@ -23,14 +23,14 @@
* http://www.mozilla.org/MPL/ * * http://www.mozilla.org/MPL/ *
***************************************************************************/ ***************************************************************************/
#include <tpropertymap.h> #include "tpicturemap.h"
#include "tpropertymap.h"
#include "asftag.h" #include "asftag.h"
using namespace Strawberry_TagLib::TagLib; using namespace Strawberry_TagLib::TagLib;
class ASF::Tag::TagPrivate class ASF::Tag::TagPrivate {
{ public:
public:
String title; String title;
String artist; String artist;
String copyright; String copyright;
@@ -39,258 +39,399 @@ public:
AttributeListMap attributeListMap; AttributeListMap attributeListMap;
}; };
ASF::Tag::Tag() : ASF::Tag::Tag() : d(new TagPrivate()) {}
Strawberry_TagLib::TagLib::Tag(),
d(new TagPrivate())
{
}
ASF::Tag::~Tag() ASF::Tag::~Tag() {
{
delete d; delete d;
} }
String ASF::Tag::title() const String ASF::Tag::title() const {
{
return d->title; return d->title;
} }
String ASF::Tag::artist() const String ASF::Tag::artist() const {
{
return d->artist; return d->artist;
} }
String ASF::Tag::album() const String ASF::Tag::album() const {
{
if(d->attributeListMap.contains("WM/AlbumTitle")) if (d->attributeListMap.contains("WM/AlbumTitle"))
return d->attributeListMap["WM/AlbumTitle"][0].toString(); return d->attributeListMap["WM/AlbumTitle"][0].toString();
return String(); return String();
} }
String ASF::Tag::copyright() const String ASF::Tag::copyright() const {
{
return d->copyright; return d->copyright;
} }
String ASF::Tag::comment() const String ASF::Tag::comment() const {
{
return d->comment; return d->comment;
} }
String ASF::Tag::rating() const String ASF::Tag::rating() const {
{
return d->rating; return d->rating;
} }
unsigned int ASF::Tag::year() const unsigned int ASF::Tag::year() const {
{
if(d->attributeListMap.contains("WM/Year")) if (d->attributeListMap.contains("WM/Year"))
return d->attributeListMap["WM/Year"][0].toString().toInt(); return d->attributeListMap["WM/Year"][0].toString().toInt();
return 0; return 0;
} }
unsigned int ASF::Tag::track() const unsigned int ASF::Tag::track() const {
{
if(d->attributeListMap.contains("WM/TrackNumber")) { if (d->attributeListMap.contains("WM/TrackNumber")) {
const ASF::Attribute attr = d->attributeListMap["WM/TrackNumber"][0]; const ASF::Attribute attr = d->attributeListMap["WM/TrackNumber"][0];
if(attr.type() == ASF::Attribute::DWordType) if (attr.type() == ASF::Attribute::DWordType)
return attr.toUInt(); return attr.toUInt();
else else
return attr.toString().toInt(); return attr.toString().toInt();
} }
if(d->attributeListMap.contains("WM/Track")) if (d->attributeListMap.contains("WM/Track"))
return d->attributeListMap["WM/Track"][0].toUInt(); return d->attributeListMap["WM/Track"][0].toUInt();
return 0; return 0;
} }
String ASF::Tag::genre() const String ASF::Tag::genre() const {
{
if(d->attributeListMap.contains("WM/Genre")) if (d->attributeListMap.contains("WM/Genre"))
return d->attributeListMap["WM/Genre"][0].toString(); return d->attributeListMap["WM/Genre"][0].toString();
return String(); return String();
} }
void ASF::Tag::setTitle(const String &value) PictureMap ASF::Tag::pictures() const {
{
PictureMap map;
if (d->attributeListMap.contains("WM/Picture")) {
AttributeList list = d->attributeListMap["WM/Picture"];
for (AttributeList::ConstIterator it = list.begin(); it != list.end(); ++it) {
ASF::Picture asfPicture = (*it).toPicture();
TagLib::Picture::Type type;
switch (asfPicture.type()) {
case ASF::Picture::FileIcon:
type = TagLib::Picture::FileIcon;
break;
case ASF::Picture::OtherFileIcon:
type = TagLib::Picture::OtherFileIcon;
break;
case ASF::Picture::FrontCover:
type = TagLib::Picture::FrontCover;
break;
case ASF::Picture::BackCover:
type = TagLib::Picture::BackCover;
break;
case ASF::Picture::LeafletPage:
type = TagLib::Picture::LeafletPage;
break;
case ASF::Picture::Media:
type = TagLib::Picture::Media;
break;
case ASF::Picture::LeadArtist:
type = TagLib::Picture::LeadArtist;
break;
case ASF::Picture::Artist:
type = TagLib::Picture::Artist;
break;
case ASF::Picture::Conductor:
type = TagLib::Picture::Conductor;
break;
case ASF::Picture::Band:
type = TagLib::Picture::Band;
break;
case ASF::Picture::Composer:
type = TagLib::Picture::Composer;
break;
case ASF::Picture::Lyricist:
type = TagLib::Picture::Lyricist;
break;
case ASF::Picture::RecordingLocation:
type = TagLib::Picture::RecordingLocation;
break;
case ASF::Picture::DuringRecording:
type = TagLib::Picture::DuringRecording;
break;
case ASF::Picture::DuringPerformance:
type = TagLib::Picture::DuringPerformance;
break;
case ASF::Picture::MovieScreenCapture:
type = TagLib::Picture::MovieScreenCapture;
break;
case ASF::Picture::ColouredFish:
type = TagLib::Picture::ColouredFish;
break;
case ASF::Picture::Illustration:
type = TagLib::Picture::Illustration;
break;
case ASF::Picture::BandLogo:
type = TagLib::Picture::BandLogo;
break;
case ASF::Picture::PublisherLogo:
type = TagLib::Picture::PublisherLogo;
break;
default:
type = TagLib::Picture::Other;
break;
}
TagLib::Picture picture(asfPicture.picture(), type, asfPicture.mimeType(), asfPicture.description());
map.insert(picture);
}
}
return PictureMap(map);
}
void ASF::Tag::setTitle(const String &value) {
d->title = value; d->title = value;
} }
void ASF::Tag::setArtist(const String &value) void ASF::Tag::setArtist(const String &value) {
{
d->artist = value; d->artist = value;
} }
void ASF::Tag::setCopyright(const String &value) void ASF::Tag::setCopyright(const String &value) {
{
d->copyright = value; d->copyright = value;
} }
void ASF::Tag::setComment(const String &value) void ASF::Tag::setComment(const String &value) {
{
d->comment = value; d->comment = value;
} }
void ASF::Tag::setRating(const String &value) void ASF::Tag::setRating(const String &value) {
{
d->rating = value; d->rating = value;
} }
void ASF::Tag::setAlbum(const String &value) void ASF::Tag::setAlbum(const String &value) {
{
setAttribute("WM/AlbumTitle", value); setAttribute("WM/AlbumTitle", value);
} }
void ASF::Tag::setGenre(const String &value) void ASF::Tag::setGenre(const String &value) {
{
setAttribute("WM/Genre", value); setAttribute("WM/Genre", value);
} }
void ASF::Tag::setYear(unsigned int value) void ASF::Tag::setYear(unsigned int value) {
{
setAttribute("WM/Year", String::number(value)); setAttribute("WM/Year", String::number(value));
} }
void ASF::Tag::setTrack(unsigned int value) void ASF::Tag::setTrack(unsigned int value) {
{
setAttribute("WM/TrackNumber", String::number(value)); setAttribute("WM/TrackNumber", String::number(value));
} }
ASF::AttributeListMap& ASF::Tag::attributeListMap() void ASF::Tag::setPictures(const PictureMap &l) {
{
removeItem("WM/Picture");
for (PictureMap::ConstIterator pictureMapIt = l.begin(); pictureMapIt != l.end(); ++pictureMapIt) {
PictureList list = pictureMapIt->second;
for (PictureList::ConstIterator pictureListIt = list.begin(); pictureListIt != list.end(); ++pictureListIt) {
const TagLib::Picture picture = (*pictureListIt);
ASF::Picture asfPicture;
asfPicture.setPicture(picture.data());
asfPicture.setMimeType(picture.mime());
asfPicture.setDescription(picture.description());
switch (picture.type()) {
case TagLib::Picture::Other:
asfPicture.setType(ASF::Picture::Other);
break;
case TagLib::Picture::FileIcon:
asfPicture.setType(ASF::Picture::FileIcon);
break;
case TagLib::Picture::OtherFileIcon:
asfPicture.setType(ASF::Picture::OtherFileIcon);
break;
case TagLib::Picture::FrontCover:
asfPicture.setType(ASF::Picture::FrontCover);
break;
case TagLib::Picture::BackCover:
asfPicture.setType(ASF::Picture::BackCover);
break;
case TagLib::Picture::LeafletPage:
asfPicture.setType(ASF::Picture::LeafletPage);
break;
case TagLib::Picture::Media:
asfPicture.setType(ASF::Picture::Media);
break;
case TagLib::Picture::LeadArtist:
asfPicture.setType(ASF::Picture::LeadArtist);
break;
case TagLib::Picture::Artist:
asfPicture.setType(ASF::Picture::Artist);
break;
case TagLib::Picture::Conductor:
asfPicture.setType(ASF::Picture::Conductor);
break;
case TagLib::Picture::Band:
asfPicture.setType(ASF::Picture::Band);
break;
case TagLib::Picture::Composer:
asfPicture.setType(ASF::Picture::Composer);
break;
case TagLib::Picture::Lyricist:
asfPicture.setType(ASF::Picture::Lyricist);
break;
case TagLib::Picture::RecordingLocation:
asfPicture.setType(ASF::Picture::RecordingLocation);
break;
case TagLib::Picture::DuringRecording:
asfPicture.setType(ASF::Picture::DuringRecording);
break;
case TagLib::Picture::DuringPerformance:
asfPicture.setType(ASF::Picture::DuringPerformance);
break;
case TagLib::Picture::MovieScreenCapture:
asfPicture.setType(ASF::Picture::MovieScreenCapture);
break;
case TagLib::Picture::ColouredFish:
asfPicture.setType(ASF::Picture::ColouredFish);
break;
case TagLib::Picture::Illustration:
asfPicture.setType(ASF::Picture::Illustration);
break;
case TagLib::Picture::BandLogo:
asfPicture.setType(ASF::Picture::BandLogo);
break;
case TagLib::Picture::PublisherLogo:
asfPicture.setType(ASF::Picture::PublisherLogo);
break;
}
addAttribute("WM/Picture", Attribute(asfPicture));
}
}
}
const ASF::AttributeListMap &ASF::Tag::attributeListMap() const {
return d->attributeListMap; return d->attributeListMap;
} }
const ASF::AttributeListMap &ASF::Tag::attributeListMap() const bool ASF::Tag::contains(const String &key) const {
{
return d->attributeListMap;
}
bool ASF::Tag::contains(const String &key) const
{
return d->attributeListMap.contains(key); return d->attributeListMap.contains(key);
} }
void ASF::Tag::removeItem(const String &key) void ASF::Tag::removeItem(const String &key) {
{
d->attributeListMap.erase(key); d->attributeListMap.erase(key);
} }
ASF::AttributeList ASF::Tag::attribute(const String &name) const ASF::AttributeList ASF::Tag::attribute(const String &name) const {
{
return d->attributeListMap[name]; return d->attributeListMap[name];
} }
void ASF::Tag::setAttribute(const String &name, const Attribute &attribute) void ASF::Tag::setAttribute(const String &name, const Attribute &attribute) {
{
AttributeList value; AttributeList value;
value.append(attribute); value.append(attribute);
d->attributeListMap.insert(name, value); d->attributeListMap.insert(name, value);
} }
void ASF::Tag::setAttribute(const String &name, const AttributeList &values) void ASF::Tag::setAttribute(const String &name, const AttributeList &values) {
{
d->attributeListMap.insert(name, values); d->attributeListMap.insert(name, values);
} }
void ASF::Tag::addAttribute(const String &name, const Attribute &attribute) void ASF::Tag::addAttribute(const String &name, const Attribute &attribute) {
{
if(d->attributeListMap.contains(name)) { if (d->attributeListMap.contains(name)) {
d->attributeListMap[name].append(attribute); d->attributeListMap[name].append(attribute);
} }
else { else {
setAttribute(name, attribute); setAttribute(name, attribute);
} }
} }
bool ASF::Tag::isEmpty() const bool ASF::Tag::isEmpty() const {
{
return Strawberry_TagLib::TagLib::Tag::isEmpty() && return Strawberry_TagLib::TagLib::Tag::isEmpty() &&
copyright().isEmpty() && copyright().isEmpty() &&
rating().isEmpty() && rating().isEmpty() &&
d->attributeListMap.isEmpty(); d->attributeListMap.isEmpty();
} }
namespace namespace {
{ const char *keyTranslation[][2] = {
const char *keyTranslation[][2] = { { "WM/AlbumTitle", "ALBUM" },
{ "WM/AlbumTitle", "ALBUM" }, { "WM/AlbumArtist", "ALBUMARTIST" },
{ "WM/AlbumArtist", "ALBUMARTIST" }, { "WM/Composer", "COMPOSER" },
{ "WM/Composer", "COMPOSER" }, { "WM/Writer", "WRITER" },
{ "WM/Writer", "WRITER" }, { "WM/Conductor", "CONDUCTOR" },
{ "WM/Conductor", "CONDUCTOR" }, { "WM/ModifiedBy", "REMIXER" },
{ "WM/ModifiedBy", "REMIXER" }, { "WM/Year", "DATE" },
{ "WM/Year", "DATE" }, { "WM/OriginalReleaseYear", "ORIGINALDATE" },
{ "WM/OriginalReleaseYear", "ORIGINALDATE" }, { "WM/Producer", "PRODUCER" },
{ "WM/Producer", "PRODUCER" }, { "WM/ContentGroupDescription", "GROUPING" },
{ "WM/ContentGroupDescription", "GROUPING" }, { "WM/SubTitle", "SUBTITLE" },
{ "WM/SubTitle", "SUBTITLE" }, { "WM/SetSubTitle", "DISCSUBTITLE" },
{ "WM/SetSubTitle", "DISCSUBTITLE" }, { "WM/TrackNumber", "TRACKNUMBER" },
{ "WM/TrackNumber", "TRACKNUMBER" }, { "WM/PartOfSet", "DISCNUMBER" },
{ "WM/PartOfSet", "DISCNUMBER" }, { "WM/Genre", "GENRE" },
{ "WM/Genre", "GENRE" }, { "WM/BeatsPerMinute", "BPM" },
{ "WM/BeatsPerMinute", "BPM" }, { "WM/Mood", "MOOD" },
{ "WM/Mood", "MOOD" }, { "WM/ISRC", "ISRC" },
{ "WM/ISRC", "ISRC" }, { "WM/Lyrics", "LYRICS" },
{ "WM/Lyrics", "LYRICS" }, { "WM/Media", "MEDIA" },
{ "WM/Media", "MEDIA" }, { "WM/Publisher", "LABEL" },
{ "WM/Publisher", "LABEL" }, { "WM/CatalogNo", "CATALOGNUMBER" },
{ "WM/CatalogNo", "CATALOGNUMBER" }, { "WM/Barcode", "BARCODE" },
{ "WM/Barcode", "BARCODE" }, { "WM/EncodedBy", "ENCODEDBY" },
{ "WM/EncodedBy", "ENCODEDBY" }, { "WM/AlbumSortOrder", "ALBUMSORT" },
{ "WM/AlbumSortOrder", "ALBUMSORT" }, { "WM/AlbumArtistSortOrder", "ALBUMARTISTSORT" },
{ "WM/AlbumArtistSortOrder", "ALBUMARTISTSORT" }, { "WM/ArtistSortOrder", "ARTISTSORT" },
{ "WM/ArtistSortOrder", "ARTISTSORT" }, { "WM/TitleSortOrder", "TITLESORT" },
{ "WM/TitleSortOrder", "TITLESORT" }, { "WM/Script", "SCRIPT" },
{ "WM/Script", "SCRIPT" }, { "WM/Language", "LANGUAGE" },
{ "WM/Language", "LANGUAGE" }, { "MusicBrainz/Track Id", "MUSICBRAINZ_TRACKID" },
{ "MusicBrainz/Track Id", "MUSICBRAINZ_TRACKID" }, { "MusicBrainz/Artist Id", "MUSICBRAINZ_ARTISTID" },
{ "MusicBrainz/Artist Id", "MUSICBRAINZ_ARTISTID" }, { "MusicBrainz/Album Id", "MUSICBRAINZ_ALBUMID" },
{ "MusicBrainz/Album Id", "MUSICBRAINZ_ALBUMID" }, { "MusicBrainz/Album Artist Id", "MUSICBRAINZ_ALBUMARTISTID" },
{ "MusicBrainz/Album Artist Id", "MUSICBRAINZ_ALBUMARTISTID" }, { "MusicBrainz/Release Group Id", "MUSICBRAINZ_RELEASEGROUPID" },
{ "MusicBrainz/Release Group Id", "MUSICBRAINZ_RELEASEGROUPID" }, { "MusicBrainz/Work Id", "MUSICBRAINZ_WORKID" },
{ "MusicBrainz/Work Id", "MUSICBRAINZ_WORKID" }, { "MusicIP/PUID", "MUSICIP_PUID" },
{ "MusicIP/PUID", "MUSICIP_PUID" }, { "Acoustid/Id", "ACOUSTID_ID" },
{ "Acoustid/Id", "ACOUSTID_ID" }, { "Acoustid/Fingerprint", "ACOUSTID_FINGERPRINT" },
{ "Acoustid/Fingerprint", "ACOUSTID_FINGERPRINT" }, };
}; const size_t keyTranslationSize = sizeof(keyTranslation) / sizeof(keyTranslation[0]);
const size_t keyTranslationSize = sizeof(keyTranslation) / sizeof(keyTranslation[0]);
String translateKey(const String &key) String translateKey(const String &key) {
{
for(size_t i = 0; i < keyTranslationSize; ++i) {
if(key == keyTranslation[i][0])
return keyTranslation[i][1];
}
return String(); for (size_t i = 0; i < keyTranslationSize; ++i) {
if (key == keyTranslation[i][0])
return keyTranslation[i][1];
} }
}
PropertyMap ASF::Tag::properties() const return String();
{
}
} // namespace
PropertyMap ASF::Tag::properties() const {
PropertyMap props; PropertyMap props;
if(!d->title.isEmpty()) { if (!d->title.isEmpty()) {
props["TITLE"] = d->title; props["TITLE"] = d->title;
} }
if(!d->artist.isEmpty()) { if (!d->artist.isEmpty()) {
props["ARTIST"] = d->artist; props["ARTIST"] = d->artist;
} }
if(!d->copyright.isEmpty()) { if (!d->copyright.isEmpty()) {
props["COPYRIGHT"] = d->copyright; props["COPYRIGHT"] = d->copyright;
} }
if(!d->comment.isEmpty()) { if (!d->comment.isEmpty()) {
props["COMMENT"] = d->comment; props["COMMENT"] = d->comment;
} }
ASF::AttributeListMap::ConstIterator it = d->attributeListMap.begin(); ASF::AttributeListMap::ConstIterator it = d->attributeListMap.begin();
for(; it != d->attributeListMap.end(); ++it) { for (; it != d->attributeListMap.end(); ++it) {
const String key = translateKey(it->first); const String key = translateKey(it->first);
if(!key.isEmpty()) { if (!key.isEmpty()) {
AttributeList::ConstIterator it2 = it->second.begin(); AttributeList::ConstIterator it2 = it->second.begin();
for(; it2 != it->second.end(); ++it2) { for (; it2 != it->second.end(); ++it2) {
if(key == "TRACKNUMBER") { if (key == "TRACKNUMBER") {
if(it2->type() == ASF::Attribute::DWordType) if (it2->type() == ASF::Attribute::DWordType)
props.insert(key, String::number(it2->toUInt())); props.insert(key, String::number(it2->toUInt()));
else else
props.insert(key, it2->toString()); props.insert(key, it2->toString());
@@ -305,39 +446,41 @@ PropertyMap ASF::Tag::properties() const
} }
} }
return props; return props;
} }
void ASF::Tag::removeUnsupportedProperties(const StringList &props) void ASF::Tag::removeUnsupportedProperties(const StringList &props) {
{
StringList::ConstIterator it = props.begin(); StringList::ConstIterator it = props.begin();
for(; it != props.end(); ++it) for (; it != props.end(); ++it)
d->attributeListMap.erase(*it); d->attributeListMap.erase(*it);
} }
PropertyMap ASF::Tag::setProperties(const PropertyMap &props) PropertyMap ASF::Tag::setProperties(const PropertyMap &props) {
{
static Map<String, String> reverseKeyMap; static Map<String, String> reverseKeyMap;
if(reverseKeyMap.isEmpty()) { if (reverseKeyMap.isEmpty()) {
int numKeys = sizeof(keyTranslation) / sizeof(keyTranslation[0]); int numKeys = sizeof(keyTranslation) / sizeof(keyTranslation[0]);
for(int i = 0; i < numKeys; i++) { for (int i = 0; i < numKeys; i++) {
reverseKeyMap[keyTranslation[i][1]] = keyTranslation[i][0]; reverseKeyMap[keyTranslation[i][1]] = keyTranslation[i][0];
} }
} }
PropertyMap origProps = properties(); PropertyMap origProps = properties();
PropertyMap::ConstIterator it = origProps.begin(); PropertyMap::ConstIterator it = origProps.begin();
for(; it != origProps.end(); ++it) { for (; it != origProps.end(); ++it) {
if(!props.contains(it->first) || props[it->first].isEmpty()) { if (!props.contains(it->first) || props[it->first].isEmpty()) {
if(it->first == "TITLE") { if (it->first == "TITLE") {
d->title.clear(); d->title.clear();
} }
else if(it->first == "ARTIST") { else if (it->first == "ARTIST") {
d->artist.clear(); d->artist.clear();
} }
else if(it->first == "COMMENT") { else if (it->first == "COMMENT") {
d->comment.clear(); d->comment.clear();
} }
else if(it->first == "COPYRIGHT") { else if (it->first == "COPYRIGHT") {
d->copyright.clear(); d->copyright.clear();
} }
else { else {
@@ -348,25 +491,25 @@ PropertyMap ASF::Tag::setProperties(const PropertyMap &props)
PropertyMap ignoredProps; PropertyMap ignoredProps;
it = props.begin(); it = props.begin();
for(; it != props.end(); ++it) { for (; it != props.end(); ++it) {
if(reverseKeyMap.contains(it->first)) { if (reverseKeyMap.contains(it->first)) {
String name = reverseKeyMap[it->first]; String name = reverseKeyMap[it->first];
removeItem(name); removeItem(name);
StringList::ConstIterator it2 = it->second.begin(); StringList::ConstIterator it2 = it->second.begin();
for(; it2 != it->second.end(); ++it2) { for (; it2 != it->second.end(); ++it2) {
addAttribute(name, *it2); addAttribute(name, *it2);
} }
} }
else if(it->first == "TITLE") { else if (it->first == "TITLE") {
d->title = it->second.toString(); d->title = it->second.toString();
} }
else if(it->first == "ARTIST") { else if (it->first == "ARTIST") {
d->artist = it->second.toString(); d->artist = it->second.toString();
} }
else if(it->first == "COMMENT") { else if (it->first == "COMMENT") {
d->comment = it->second.toString(); d->comment = it->second.toString();
} }
else if(it->first == "COPYRIGHT") { else if (it->first == "COPYRIGHT") {
d->copyright = it->second.toString(); d->copyright = it->second.toString();
} }
else { else {
@@ -375,4 +518,5 @@ PropertyMap ASF::Tag::setProperties(const PropertyMap &props)
} }
return ignoredProps; return ignoredProps;
} }

View File

@@ -35,177 +35,167 @@
namespace Strawberry_TagLib { namespace Strawberry_TagLib {
namespace TagLib { namespace TagLib {
namespace ASF { namespace ASF {
typedef List<Attribute> AttributeList; typedef List<Attribute> AttributeList;
typedef Map<String, AttributeList> AttributeListMap; typedef Map<String, AttributeList> AttributeListMap;
class TAGLIB_EXPORT Tag : public Strawberry_TagLib::TagLib::Tag { class TAGLIB_EXPORT Tag : public Strawberry_TagLib::TagLib::Tag {
friend class File; friend class File;
public: public:
explicit Tag();
Tag(); ~Tag() override;
virtual ~Tag(); /*!
* Returns the track name.
*/
String title() const override;
/*! /*!
* Returns the track name. * Returns the artist name.
*/ */
virtual String title() const; String artist() const override;
/*! /*!
* Returns the artist name. * Returns the album name; if no album name is present in the tag String::null will be returned.
*/ */
virtual String artist() const; String album() const override;
/*! /*!
* Returns the album name; if no album name is present in the tag * Returns the track comment.
* String::null will be returned. */
*/ String comment() const override;
virtual String album() const;
/*! /*!
* Returns the track comment. * Returns the genre name; if no genre is present in the tag String::null will be returned.
*/ */
virtual String comment() const; String genre() const override;
/*! /*!
* Returns the genre name; if no genre is present in the tag String::null * Returns the rating.
* will be returned. */
*/ virtual String rating() const;
virtual String genre() const;
/*! /*!
* Returns the rating. * Returns the genre name; if no genre is present in the tag String::null will be returned.
*/ */
virtual String rating() const; virtual String copyright() const;
/*! /*!
* Returns the genre name; if no genre is present in the tag String::null * Returns the year; if there is no year set, this will return 0.
* will be returned. */
*/ unsigned int year() const override;
virtual String copyright() const;
/*! /*!
* Returns the year; if there is no year set, this will return 0. * Returns the track number; if there is no track number set, this will return 0.
*/ */
virtual unsigned int year() const; unsigned int track() const override;
/*! PictureMap pictures() const override;
* Returns the track number; if there is no track number set, this will
* return 0.
*/
virtual unsigned int track() const;
/*! /*!
* Sets the title to \a s. * Sets the title to \a s.
*/ */
virtual void setTitle(const String &s); void setTitle(const String &value) override;
/*! /*!
* Sets the artist to \a s. * Sets the artist to \a s.
*/ */
virtual void setArtist(const String &s); void setArtist(const String &value) override;
/*! /*!
* Sets the album to \a s. If \a s is String::null then this value will be * Sets the album to \a s. If \a s is String::null then this value will be cleared.
* cleared. */
*/ void setAlbum(const String &value) override;
virtual void setAlbum(const String &s);
/*! /*!
* Sets the comment to \a s. * Sets the comment to \a s.
*/ */
virtual void setComment(const String &s); void setComment(const String &value) override;
/*! /*!
* Sets the rating to \a s. * Sets the rating to \a s.
*/ */
virtual void setRating(const String &s); virtual void setRating(const String &value);
/*! /*!
* Sets the copyright to \a s. * Sets the copyright to \a s.
*/ */
virtual void setCopyright(const String &s); virtual void setCopyright(const String &value);
/*! /*!
* Sets the genre to \a s. * Sets the genre to \a s.
*/ */
virtual void setGenre(const String &s); void setGenre(const String &value) override;
/*! /*!
* Sets the year to \a i. If \a s is 0 then this value will be cleared. * Sets the year to \a i. If \a s is 0 then this value will be cleared.
*/ */
virtual void setYear(unsigned int i); void setYear(unsigned int value) override;
/*! /*!
* Sets the track to \a i. If \a s is 0 then this value will be cleared. * Sets the track to \a i. If \a s is 0 then this value will be cleared.
*/ */
virtual void setTrack(unsigned int i); void setTrack(unsigned int value) override;
/*! void setPictures(const PictureMap&) override;
* Returns true if the tag does not contain any data. This should be
* reimplemented in subclasses that provide more than the basic tagging
* abilities in this class.
*/
virtual bool isEmpty() const;
/*! /*!
* \deprecated * Returns true if the tag does not contain any data.
*/ * This should be reimplemented in subclasses that provide more than the basic tagging abilities in this class.
AttributeListMap &attributeListMap(); */
bool isEmpty() const override;
/*! /*!
* Returns a reference to the item list map. This is an AttributeListMap of * Returns a reference to the item list map. This is an AttributeListMap of all of the items in the tag.
* all of the items in the tag. */
*/ const AttributeListMap &attributeListMap() const;
const AttributeListMap &attributeListMap() const;
/*! /*!
* \return True if a value for \a attribute is currently set. * \return True if a value for \a attribute is currently set.
*/ */
bool contains(const String &name) const; bool contains(const String &key) const;
/*! /*!
* Removes the \a key attribute from the tag * Removes the \a key attribute from the tag
*/ */
void removeItem(const String &name); void removeItem(const String &key);
/*! /*!
* \return The list of values for the key \a name, or an empty list if no * \return The list of values for the key \a name, or an empty list if no values have been set.
* values have been set. */
*/ AttributeList attribute(const String &name) const;
AttributeList attribute(const String &name) const;
/*! /*!
* Sets the \a key attribute to the value of \a attribute. If an attribute * Sets the \a key attribute to the value of \a attribute. If an attribute with the \a key is already present, it will be replaced.
* with the \a key is already present, it will be replaced. */
*/ void setAttribute(const String &name, const Attribute &attribute);
void setAttribute(const String &name, const Attribute &attribute);
/*! /*!
* Sets multiple \a values to the key \a name. * Sets multiple \a values to the key \a name.
*/ */
void setAttribute(const String &name, const AttributeList &values); void setAttribute(const String &name, const AttributeList &values);
/*! /*!
* Sets the \a key attribute to the value of \a attribute. If an attribute * Sets the \a key attribute to the value of \a attribute. If an attribute
* with the \a key is already present, it will be added to the list. * with the \a key is already present, it will be added to the list.
*/ */
void addAttribute(const String &name, const Attribute &attribute); void addAttribute(const String &name, const Attribute &attribute);
PropertyMap properties() const; PropertyMap properties() const override;
void removeUnsupportedProperties(const StringList& properties); void removeUnsupportedProperties(const StringList &props) override;
PropertyMap setProperties(const PropertyMap &properties); PropertyMap setProperties(const PropertyMap &props) override;
private: private:
class TagPrivate;
TagPrivate *d;
};
} // namespace ASF
} // namespace TagLib
} // namespace Strawberry_TagLib
class TagPrivate;
TagPrivate *d;
};
}
}
}
#endif #endif

View File

@@ -31,76 +31,70 @@
#ifndef DO_NOT_DOCUMENT // tell Doxygen not to document this header #ifndef DO_NOT_DOCUMENT // tell Doxygen not to document this header
namespace Strawberry_TagLib { namespace Strawberry_TagLib {
namespace TagLib namespace TagLib {
{ namespace ASF {
namespace ASF namespace {
{
namespace
{
inline unsigned short readWORD(File *file, bool *ok = 0) inline unsigned short readWORD(File *file, bool *ok = nullptr) {
{ const ByteVector v = file->readBlock(2);
const ByteVector v = file->readBlock(2); if (v.size() != 2) {
if(v.size() != 2) { if (ok) *ok = false;
if(ok) *ok = false; return 0;
return 0;
}
if(ok) *ok = true;
return v.toUShort(false);
}
inline unsigned int readDWORD(File *file, bool *ok = 0)
{
const ByteVector v = file->readBlock(4);
if(v.size() != 4) {
if(ok) *ok = false;
return 0;
}
if(ok) *ok = true;
return v.toUInt(false);
}
inline long long readQWORD(File *file, bool *ok = 0)
{
const ByteVector v = file->readBlock(8);
if(v.size() != 8) {
if(ok) *ok = false;
return 0;
}
if(ok) *ok = true;
return v.toLongLong(false);
}
inline String readString(File *file, int length)
{
ByteVector data = file->readBlock(length);
unsigned int size = data.size();
while (size >= 2) {
if(data[size - 1] != '\0' || data[size - 2] != '\0') {
break;
}
size -= 2;
}
if(size != data.size()) {
data.resize(size);
}
return String(data, String::UTF16LE);
}
inline ByteVector renderString(const String &str, bool includeLength = false)
{
ByteVector data = str.data(String::UTF16LE) + ByteVector::fromShort(0, false);
if(includeLength) {
data = ByteVector::fromShort(data.size(), false) + data;
}
return data;
}
}
} }
if (ok) *ok = true;
return v.toUInt16LE(0);
} }
inline unsigned int readDWORD(File *file, bool *ok = nullptr) {
const ByteVector v = file->readBlock(4);
if (v.size() != 4) {
if (ok) *ok = false;
return 0;
}
if (ok) *ok = true;
return v.toUInt32LE(0);
} }
inline long long readQWORD(File *file, bool *ok = nullptr) {
const ByteVector v = file->readBlock(8);
if (v.size() != 8) {
if (ok) *ok = false;
return 0;
}
if (ok) *ok = true;
return v.toInt64LE(0);
}
inline String readString(File *file, int length) {
ByteVector data = file->readBlock(length);
size_t size = data.size();
while (size >= 2) {
if (data[size - 1] != '\0' || data[size - 2] != '\0') {
break;
}
size -= 2;
}
if (size != data.size()) {
data.resize(size);
}
return String(data, String::UTF16LE);
}
inline ByteVector renderString(const String &str, bool includeLength = false) {
ByteVector data = str.data(String::UTF16LE) + ByteVector::fromUInt16LE(0);
if (includeLength) {
data = ByteVector::fromUInt16LE(data.size()) + data;
}
return data;
}
} // namespace
} // namespace ASF
} // namespace TagLib
} // namespace Strawberry_TagLib
#endif #endif
#endif #endif

View File

@@ -23,95 +23,31 @@
* http://www.mozilla.org/MPL/ * * http://www.mozilla.org/MPL/ *
***************************************************************************/ ***************************************************************************/
#include <tbytevector.h> #include "tstringlist.h"
#include "aiffproperties.h"
#include "apeproperties.h"
#include "asfproperties.h"
#include "flacproperties.h"
#include "mp4properties.h"
#include "mpcproperties.h"
#include "mpegproperties.h"
#include "opusproperties.h"
#include "speexproperties.h"
#include "trueaudioproperties.h"
#include "vorbisproperties.h"
#include "wavproperties.h"
#include "wavpackproperties.h"
#include "dsfproperties.h"
#include "dsdiffproperties.h"
#include "audioproperties.h" #include "audioproperties.h"
using namespace Strawberry_TagLib::TagLib; using namespace Strawberry_TagLib::TagLib;
// This macro is a workaround for the fact that we can't add virtual functions.
// Should be true virtual functions in taglib2.
#define VIRTUAL_FUNCTION_WORKAROUND(function_name, default_value) \
if(dynamic_cast<const APE::Properties*>(this)) \
return dynamic_cast<const APE::Properties*>(this)->function_name(); \
else if(dynamic_cast<const ASF::Properties*>(this)) \
return dynamic_cast<const ASF::Properties*>(this)->function_name(); \
else if(dynamic_cast<const FLAC::Properties*>(this)) \
return dynamic_cast<const FLAC::Properties*>(this)->function_name(); \
else if(dynamic_cast<const MP4::Properties*>(this)) \
return dynamic_cast<const MP4::Properties*>(this)->function_name(); \
else if(dynamic_cast<const MPC::Properties*>(this)) \
return dynamic_cast<const MPC::Properties*>(this)->function_name(); \
else if(dynamic_cast<const MPEG::Properties*>(this)) \
return dynamic_cast<const MPEG::Properties*>(this)->function_name(); \
else if(dynamic_cast<const Ogg::Opus::Properties*>(this)) \
return dynamic_cast<const Ogg::Opus::Properties*>(this)->function_name(); \
else if(dynamic_cast<const Ogg::Speex::Properties*>(this)) \
return dynamic_cast<const Ogg::Speex::Properties*>(this)->function_name(); \
else if(dynamic_cast<const TrueAudio::Properties*>(this)) \
return dynamic_cast<const TrueAudio::Properties*>(this)->function_name(); \
else if(dynamic_cast<const RIFF::AIFF::Properties*>(this)) \
return dynamic_cast<const RIFF::AIFF::Properties*>(this)->function_name(); \
else if(dynamic_cast<const RIFF::WAV::Properties*>(this)) \
return dynamic_cast<const RIFF::WAV::Properties*>(this)->function_name(); \
else if(dynamic_cast<const Vorbis::Properties*>(this)) \
return dynamic_cast<const Vorbis::Properties*>(this)->function_name(); \
else if(dynamic_cast<const WavPack::Properties*>(this)) \
return dynamic_cast<const WavPack::Properties*>(this)->function_name(); \
else if(dynamic_cast<const DSF::Properties*>(this)) \
return dynamic_cast<const DSF::Properties*>(this)->function_name(); \
else if(dynamic_cast<const DSDIFF::Properties*>(this)) \
return dynamic_cast<const DSDIFF::Properties*>(this)->function_name(); \
else \
return (default_value);
class AudioProperties::AudioPropertiesPrivate
{
};
//////////////////////////////////////////////////////////////////////////////// ////////////////////////////////////////////////////////////////////////////////
// public methods // public methods
//////////////////////////////////////////////////////////////////////////////// ////////////////////////////////////////////////////////////////////////////////
AudioProperties::~AudioProperties() AudioProperties::~AudioProperties() {}
{
}
int AudioProperties::lengthInSeconds() const String AudioProperties::toString() const {
{
VIRTUAL_FUNCTION_WORKAROUND(lengthInSeconds, 0) StringList desc;
} desc.append("Audio");
desc.append(String::number(lengthInSeconds()) + " seconds");
desc.append(String::number(bitrate()) + " kbps");
return desc.toString(", ");
int AudioProperties::lengthInMilliseconds() const
{
VIRTUAL_FUNCTION_WORKAROUND(lengthInMilliseconds, 0)
} }
//////////////////////////////////////////////////////////////////////////////// ////////////////////////////////////////////////////////////////////////////////
// protected methods // protected methods
//////////////////////////////////////////////////////////////////////////////// ////////////////////////////////////////////////////////////////////////////////
AudioProperties::AudioProperties(ReadStyle) : AudioProperties::AudioProperties() : d(nullptr) {}
d(0)
{
}

View File

@@ -27,103 +27,94 @@
#define TAGLIB_AUDIOPROPERTIES_H #define TAGLIB_AUDIOPROPERTIES_H
#include "taglib_export.h" #include "taglib_export.h"
#include "tstring.h"
namespace Strawberry_TagLib { namespace Strawberry_TagLib {
namespace TagLib { namespace TagLib {
//! A simple, abstract interface to common audio properties //! A simple, abstract interface to common audio properties
/*! /*!
* The values here are common to most audio formats. For more specific, codec * The values here are common to most audio formats.
* dependent values, please see see the subclasses APIs. This is meant to * For more specific, codec dependent values, please see see the subclasses APIs.
* compliment the Strawberry_TagLib::TagLib::File and Strawberry_TagLib::TagLib::Tag APIs in providing a simple * This is meant to compliment the TagLib::File and TagLib::Tag APIs in providing a simple
* interface that is sufficient for most applications. * interface that is sufficient for most applications.
*/ */
class TAGLIB_EXPORT AudioProperties class TAGLIB_EXPORT AudioProperties {
{ public:
public: /*!
* Reading audio properties from a file can sometimes be very time consuming
/*! * and for the most accurate results can often involve reading the entire file.
* Reading audio properties from a file can sometimes be very time consuming * Because in many situations speed is critical or the accuracy of the values
* and for the most accurate results can often involve reading the entire * is not particularly important this allows the level of desired accuracy to be set.
* file. Because in many situations speed is critical or the accuracy of the */
* values is not particularly important this allows the level of desired enum ReadStyle {
* accuracy to be set. //! Read as little of the file as possible
*/ Fast,
enum ReadStyle { //! Read more of the file and make better values guesses
//! Read as little of the file as possible Average,
Fast, //! Read as much of the file as needed to report accurate values
//! Read more of the file and make better values guesses Accurate
Average,
//! Read as much of the file as needed to report accurate values
Accurate
};
/*!
* Destroys this AudioProperties instance.
*/
virtual ~AudioProperties();
/*!
* Returns the length of the file in seconds.
*/
virtual int length() const = 0;
/*!
* Returns the length of the file in seconds. The length is rounded down to
* the nearest whole second.
*
* \see lengthInMilliseconds()
*/
// BIC: make virtual
int lengthInSeconds() const;
/*!
* Returns the length of the file in milliseconds.
*
* \see lengthInSeconds()
*/
// BIC: make virtual
int lengthInMilliseconds() const;
/*!
* Returns the most appropriate bit rate for the file in kb/s. For constant
* bitrate formats this is simply the bitrate of the file. For variable
* bitrate formats this is either the average or nominal bitrate.
*/
virtual int bitrate() const = 0;
/*!
* Returns the sample rate in Hz.
*/
virtual int sampleRate() const = 0;
/*!
* Returns the number of audio channels.
*/
virtual int channels() const = 0;
protected:
/*!
* Construct an audio properties instance. This is protected as this class
* should not be instantiated directly, but should be instantiated via its
* subclasses and can be fetched from the FileRef or File APIs.
*
* \see ReadStyle
*/
AudioProperties(ReadStyle style);
private:
AudioProperties(const AudioProperties &);
AudioProperties &operator=(const AudioProperties &);
class AudioPropertiesPrivate;
AudioPropertiesPrivate *d;
}; };
} /*!
} * Destroys this AudioProperties instance.
*/
virtual ~AudioProperties();
/*!
* Returns the length of the file in seconds. The length is rounded down to the nearest whole second.
*
* \see lengthInMilliseconds()
*/
virtual int lengthInSeconds() const = 0;
/*!
* Returns the length of the file in milliseconds.
*
* \see lengthInSeconds()
*/
virtual int lengthInMilliseconds() const = 0;
/*!
* Returns the most appropriate bit rate for the file in kb/s. For constant bitrate formats this is simply the bitrate of the file.
* For variable bitrate formats this is either the average or nominal bitrate.
*/
virtual int bitrate() const = 0;
/*!
* Returns the sample rate in Hz.
*/
virtual int sampleRate() const = 0;
/*!
* Returns the number of audio channels.
*/
virtual int channels() const = 0;
/*!
* Returns description of the audio file.
*/
virtual String toString() const;
protected:
/*!
* Construct an audio properties instance.
* This is protected as this class should not be instantiated directly,
* but should be instantiated via its subclasses and can be fetched from the FileRef or File APIs.
*/
explicit AudioProperties();
private:
AudioProperties(const AudioProperties&);
AudioProperties &operator=(const AudioProperties&);
class AudioPropertiesPrivate;
AudioPropertiesPrivate *d;
};
} // namespace TagLib
} // namespace Strawberry_TagLib
#endif #endif

View File

@@ -26,137 +26,122 @@
#include "dsdiffdiintag.h" #include "dsdiffdiintag.h"
#include "tstringlist.h" #include "tstringlist.h"
#include "tpropertymap.h" #include "tpropertymap.h"
#include "tpicturemap.h"
using namespace Strawberry_TagLib::TagLib; using namespace Strawberry_TagLib::TagLib;
using namespace DSDIFF::DIIN; using namespace DSDIFF::DIIN;
class DSDIFF::DIIN::Tag::TagPrivate class DSDIFF::DIIN::Tag::TagPrivate {
{ public:
public: TagPrivate() {
TagPrivate()
{
} }
String title; String title;
String artist; String artist;
}; };
DSDIFF::DIIN::Tag::Tag() : Strawberry_TagLib::TagLib::Tag() DSDIFF::DIIN::Tag::Tag() {
{
d = new TagPrivate; d = new TagPrivate;
} }
DSDIFF::DIIN::Tag::~Tag() DSDIFF::DIIN::Tag::~Tag() {
{
delete d; delete d;
} }
String DSDIFF::DIIN::Tag::title() const String DSDIFF::DIIN::Tag::title() const {
{
return d->title; return d->title;
} }
String DSDIFF::DIIN::Tag::artist() const String DSDIFF::DIIN::Tag::artist() const {
{
return d->artist; return d->artist;
} }
String DSDIFF::DIIN::Tag::album() const String DSDIFF::DIIN::Tag::album() const {
{
return String(); return String();
} }
String DSDIFF::DIIN::Tag::comment() const String DSDIFF::DIIN::Tag::comment() const {
{
return String(); return String();
} }
String DSDIFF::DIIN::Tag::genre() const String DSDIFF::DIIN::Tag::genre() const {
{
return String(); return String();
} }
unsigned int DSDIFF::DIIN::Tag::year() const unsigned int DSDIFF::DIIN::Tag::year() const {
{
return 0; return 0;
} }
unsigned int DSDIFF::DIIN::Tag::track() const unsigned int DSDIFF::DIIN::Tag::track() const {
{
return 0; return 0;
} }
void DSDIFF::DIIN::Tag::setTitle(const String &title) PictureMap DSDIFF::DIIN::Tag::pictures() const {
{ return PictureMap();
if(title.isNull() || title.isEmpty())
d->title = String();
else
d->title = title;
} }
void DSDIFF::DIIN::Tag::setArtist(const String &artist) void DSDIFF::DIIN::Tag::setTitle(const String &title) {
{
if(artist.isNull() || artist.isEmpty()) d->title = title;
d->artist = String();
else
d->artist = artist;
} }
void DSDIFF::DIIN::Tag::setAlbum(const String &) void DSDIFF::DIIN::Tag::setArtist(const String &artist) {
{
d->artist = artist;
} }
void DSDIFF::DIIN::Tag::setComment(const String &) void DSDIFF::DIIN::Tag::setAlbum(const String &) {}
{
}
void DSDIFF::DIIN::Tag::setGenre(const String &) void DSDIFF::DIIN::Tag::setComment(const String &) {}
{
}
void DSDIFF::DIIN::Tag::setYear(unsigned int) void DSDIFF::DIIN::Tag::setGenre(const String &) {}
{
}
void DSDIFF::DIIN::Tag::setTrack(unsigned int) void DSDIFF::DIIN::Tag::setYear(unsigned int) {}
{
} void DSDIFF::DIIN::Tag::setTrack(unsigned int) {}
void DSDIFF::DIIN::Tag::setPictures(const PictureMap&) {}
PropertyMap DSDIFF::DIIN::Tag::properties() const {
PropertyMap DSDIFF::DIIN::Tag::properties() const
{
PropertyMap properties; PropertyMap properties;
properties["TITLE"] = d->title; properties["TITLE"] = d->title;
properties["ARTIST"] = d->artist; properties["ARTIST"] = d->artist;
return properties; return properties;
} }
PropertyMap DSDIFF::DIIN::Tag::setProperties(const PropertyMap &origProps) PropertyMap DSDIFF::DIIN::Tag::setProperties(const PropertyMap &origProps) {
{
PropertyMap properties(origProps); PropertyMap properties(origProps);
properties.removeEmpty(); properties.removeEmpty();
StringList oneValueSet; StringList oneValueSet;
if(properties.contains("TITLE")) { if (properties.contains("TITLE")) {
d->title = properties["TITLE"].front(); d->title = properties["TITLE"].front();
oneValueSet.append("TITLE"); oneValueSet.append("TITLE");
} else }
else
d->title = String(); d->title = String();
if(properties.contains("ARTIST")) { if (properties.contains("ARTIST")) {
d->artist = properties["ARTIST"].front(); d->artist = properties["ARTIST"].front();
oneValueSet.append("ARTIST"); oneValueSet.append("ARTIST");
} else }
else
d->artist = String(); d->artist = String();
// for each tag that has been set above, remove the first entry in the corresponding // for each tag that has been set above, remove the first entry in the corresponding
// value list. The others will be returned as unsupported by this format. // value list. The others will be returned as unsupported by this format.
for(StringList::Iterator it = oneValueSet.begin(); it != oneValueSet.end(); ++it) { for (StringList::Iterator it = oneValueSet.begin(); it != oneValueSet.end(); ++it) {
if(properties[*it].size() == 1) if (properties[*it].size() == 1)
properties.erase(*it); properties.erase(*it);
else else
properties[*it].erase(properties[*it].begin()); properties[*it].erase(properties[*it].begin());
} }
return properties; return properties;
}
}

View File

@@ -30,123 +30,125 @@
namespace Strawberry_TagLib { namespace Strawberry_TagLib {
namespace TagLib { namespace TagLib {
namespace DSDIFF {
namespace DIIN {
namespace DSDIFF { /*!
* Tags from the Edited Master Chunk Info
*
* Only Title and Artist tags are supported
*/
namespace DIIN { class TAGLIB_EXPORT Tag : public Strawberry_TagLib::TagLib::Tag {
public:
explicit Tag();
~Tag() override;
/*! /*!
* Tags from the Edited Master Chunk Info * Returns the track name; if no track name is present in the tag String() will be returned.
* */
* Only Title and Artist tags are supported String title() const override;
*/
class TAGLIB_EXPORT Tag : public Strawberry_TagLib::TagLib::Tag
{
public:
Tag();
virtual ~Tag();
/*! /*!
* Returns the track name; if no track name is present in the tag * Returns the artist name; if no artist name is present in the tag String() will be returned.
* String() will be returned. */
*/ String artist() const override;
String title() const;
/*! /*!
* Returns the artist name; if no artist name is present in the tag * Not supported. Therefore always returns String().
* String() will be returned. */
*/ String album() const override;
String artist() const;
/*! /*!
* Not supported. Therefore always returns String(). * Not supported. Therefore always returns String().
*/ */
String album() const; String comment() const override;
/*! /*!
* Not supported. Therefore always returns String(). * Not supported. Therefore always returns String().
*/ */
String comment() const; String genre() const override;
/*! /*!
* Not supported. Therefore always returns String(). * Not supported. Therefore always returns 0.
*/ */
String genre() const; unsigned int year() const override;
/*! /*!
* Not supported. Therefore always returns 0. * Not supported. Therefore always returns 0.
*/ */
unsigned int year() const; unsigned int track() const override;
/*! /*!
* Not supported. Therefore always returns 0. * Not supported. Therefore always returns an empty list.
*/ */
unsigned int track() const; PictureMap pictures() const override;
/*! /*!
* Sets the title to \a title. If \a title is String() then this * Sets the title to \a title. If \a title is String() then this value will be cleared.
* value will be cleared. */
*/ void setTitle(const String &title) override;
void setTitle(const String &title);
/*! /*!
* Sets the artist to \a artist. If \a artist is String() then this * Sets the artist to \a artist. If \a artist is String() then this value will be cleared.
* value will be cleared. */
*/ void setArtist(const String &artist) override;
void setArtist(const String &artist);
/*! /*!
* Not supported and therefore ignored. * Not supported and therefore ignored.
*/ */
void setAlbum(const String &album); void setAlbum(const String &album) override;
/*! /*!
* Not supported and therefore ignored. * Not supported and therefore ignored.
*/ */
void setComment(const String &comment); void setComment(const String &comment) override;
/*! /*!
* Not supported and therefore ignored. * Not supported and therefore ignored.
*/ */
void setGenre(const String &genre); void setGenre(const String &genre) override;
/*! /*!
* Not supported and therefore ignored. * Not supported and therefore ignored.
*/ */
void setYear(unsigned int year); void setYear(unsigned int year) override;
/*! /*!
* Not supported and therefore ignored. * Not supported and therefore ignored.
*/ */
void setTrack(unsigned int track); void setTrack(unsigned int track) override;
/*! /*!
* Implements the unified property interface -- export function. * Not supported and therefore ignored.
* Since the DIIN tag is very limited, the exported map is as well. */
*/ void setPictures(const PictureMap&) override;
PropertyMap properties() const;
/*! /*!
* Implements the unified property interface -- import function. * Implements the unified property interface -- export function.
* Because of the limitations of the DIIN file tag, any tags besides * Since the DIIN tag is very limited, the exported map is as well.
* TITLE and ARTIST, will be */
* returned. Additionally, if the map contains tags with multiple values, PropertyMap properties() const override;
* all but the first will be contained in the returned map of unsupported
* properties.
*/
PropertyMap setProperties(const PropertyMap &);
private: /*!
Tag(const Tag &); * Implements the unified property interface -- import function.
Tag &operator=(const Tag &); * Because of the limitations of the DIIN file tag, any tags besides
* TITLE and ARTIST, will be returned.
* Additionally, if the map contains tags with multiple values,
* all but the first will be contained in the returned map of unsupported properties.
*/
PropertyMap setProperties(const PropertyMap &) override;
class TagPrivate; private:
TagPrivate *d; Tag(const Tag &);
}; Tag &operator=(const Tag &);
}
} class TagPrivate;
} TagPrivate *d;
} };
} // namespace DIIN
} // namespace DSDIFF
} // namespace TagLib
} // namespace Strawberry_TagLib
#endif #endif

File diff suppressed because it is too large Load Diff

View File

@@ -34,229 +34,232 @@
namespace Strawberry_TagLib { namespace Strawberry_TagLib {
namespace TagLib { namespace TagLib {
//! An implementation of DSDIFF metadata //! An implementation of DSDIFF metadata
/*!
* This is implementation of DSDIFF metadata.
*
* This supports an ID3v2 tag as well as reading stream from the ID3 RIFF
* chunk as well as properties from the file.
* Description of the DSDIFF format is available at http://dsd-guide.com/sites/default/files/white-papers/DSDIFF_1.5_Spec.pdf
* DSDIFF standard does not explicitly specify the ID3V2 chunk
* It can be found at the root level, but also sometimes inside the PROP chunk.
* In addition, title and artist info are stored as part of the standard
*/
namespace DSDIFF {
//! An implementation of TagLib::File with DSDIFF specific methods
/*!
* This implements and provides an interface for DSDIFF files to the
* TagLib::Tag and TagLib::AudioProperties interfaces by way of implementing the
* abstract TagLib::File API as well as providing some additional information specific to DSDIFF files.
*/
class TAGLIB_EXPORT File : public Strawberry_TagLib::TagLib::File {
public:
/*!
* This set of flags is used for various operations and is suitable for
* being OR-ed together.
*/
enum TagTypes {
//! Empty set. Matches no tag types.
NoTags = 0x0000,
//! Matches DIIN tags.
DIIN = 0x0002,
//! Matches ID3v1 tags.
ID3v2 = 0x0002,
//! Matches all tag types.
AllTags = 0xffff
};
/*! /*!
* This is implementation of DSDIFF metadata. * Constructs an DSDIFF file from \a file.
* If \a readProperties is true the file's audio properties will also be read.
* *
* This supports an ID3v2 tag as well as reading stream from the ID3 RIFF * \note In the current implementation, \a propertiesStyle is ignored.
* chunk as well as properties from the file.
* Description of the DSDIFF format is available
* at http://dsd-guide.com/sites/default/files/white-papers/DSDIFF_1.5_Spec.pdf
* DSDIFF standard does not explictly specify the ID3V2 chunk
* It can be found at the root level, but also sometimes inside the PROP chunk
* In addition, title and artist info are stored as part of the standard
*/ */
explicit File(FileName file, bool readProperties = true, AudioProperties::ReadStyle propertiesStyle = AudioProperties::Average);
namespace DSDIFF { /*!
* Constructs an DSDIFF file from \a stream.
* If \a readProperties is true the file's audio properties will also be read.
*
* \note TagLib will *not* take ownership of the stream, the caller is responsible for deleting it after the File object.
*
* \note In the current implementation, \a propertiesStyle is ignored.
*/
explicit File(IOStream *stream, bool readProperties = true, AudioProperties::ReadStyle propertiesStyle = AudioProperties::Average);
//! An implementation of Strawberry_TagLib::TagLib::File with DSDIFF specific methods /*!
* Destroys this instance of the File.
*/
~File() override;
/*! /*!
* This implements and provides an interface for DSDIFF files to the * Returns a pointer to a tag that is the union of the ID3v2 and DIIN tags.
* Strawberry_TagLib::TagLib::Tag and Strawberry_TagLib::TagLib::AudioProperties interfaces by way of implementing * The ID3v2 tag is given priority in reading the information -- if requested information exists in both the ID3v2 tag and the ID3v1 tag,
* the abstract Strawberry_TagLib::TagLib::File API as well as providing some additional * the information from the ID3v2 tag will be returned.
* information specific to DSDIFF files. *
*/ * If you would like more granular control over the content of the tags, with the concession of generality, use the tag-type specific calls.
*
* \note As this tag is not implemented as an ID3v2 tag or a DIIN tag,
* but a union of the two this pointer may not be cast to the specific tag types.
*
* \see ID3v2Tag()
* \see DIINTag()
*/
Tag *tag() const override;
class TAGLIB_EXPORT File : public Strawberry_TagLib::TagLib::File /*!
{ * Returns the ID3V2 Tag for this file.
public: *
/*! * \note This always returns a valid pointer regardless of whether or not the file on disk has an ID3v2 tag.
* Constructs an DSDIFF file from \a file. If \a readProperties is true * Use hasID3v2Tag() to check if the file on disk actually has an ID3v2 tag.
* the file's audio properties will also be read. *
* * \see hasID3v2Tag()
* \note In the current implementation, \a propertiesStyle is ignored. */
*/ ID3v2::Tag *ID3v2Tag(bool create = false) const;
File(FileName file, bool readProperties = true,
Properties::ReadStyle propertiesStyle = Properties::Average);
/*! /*!
* Constructs an DSDIFF file from \a stream. If \a readProperties is true * Returns the DSDIFF DIIN Tag for this file
* the file's audio properties will also be read. *
* */
* \note TagLib will *not* take ownership of the stream, the caller is DSDIFF::DIIN::Tag *DIINTag(bool create = false) const;
* responsible for deleting it after the File object.
*
* \note In the current implementation, \a propertiesStyle is ignored.
*/
File(IOStream *stream, bool readProperties = true,
Properties::ReadStyle propertiesStyle = Properties::Average);
/*! /*!
* Destroys this instance of the File. * Implements the unified property interface -- export function.
*/ * This method forwards to ID3v2::Tag::properties().
virtual ~File(); */
PropertyMap properties() const override;
/*! void removeUnsupportedProperties(const StringList &properties) override;
* Returns a pointer to a tag that is the union of the ID3v2 and DIIN
* tags. The ID3v2 tag is given priority in reading the information -- if
* requested information exists in both the ID3v2 tag and the ID3v1 tag,
* the information from the ID3v2 tag will be returned.
*
* If you would like more granular control over the content of the tags,
* with the concession of generality, use the tag-type specific calls.
*
* \note As this tag is not implemented as an ID3v2 tag or a DIIN tag,
* but a union of the two this pointer may not be cast to the specific
* tag types.
*
* \see ID3v2Tag()
* \see DIINTag()
*/
virtual Tag *tag() const;
/*! /*!
* Returns the ID3V2 Tag for this file. * Implements the unified property interface -- import function.
* * This method forwards to ID3v2::Tag::setProperties().
* \note This always returns a valid pointer regardless of whether or not */
* the file on disk has an ID3v2 tag. Use hasID3v2Tag() to check if the PropertyMap setProperties(const PropertyMap &properties) override;
* file on disk actually has an ID3v2 tag.
*
* \see hasID3v2Tag()
*/
virtual ID3v2::Tag *ID3v2Tag() const;
/*! /*!
* Returns the DSDIFF DIIN Tag for this file * Returns the AIFF::AudioProperties for this file.
* * If no audio properties were read then this will return a null pointer.
*/ */
DSDIFF::DIIN::Tag *DIINTag() const; AudioProperties *audioProperties() const override;
/*! /*!
* Implements the unified property interface -- export function. * Save the file. If at least one tag -- ID3v1 or DIIN -- exists this will duplicate its content into the other tag.
* This method forwards to ID3v2::Tag::properties(). * This returns true if saving was successful.
*/ *
PropertyMap properties() const; * If neither exists or if both tags are empty, this will strip the tags from the file.
*
* This is the same as calling save(AllTags);
*
* If you would like more granular control over the content of the tags,
* with the concession of generality, use paramaterized save call below.
*
* \see save(int tags)
*/
bool save() override;
void removeUnsupportedProperties(const StringList &properties); /*!
* Save the file. If \a strip is specified,
* it is possible to choose if tags not specified in \a tags should be stripped from the file or retained.
* With \a version, it is possible to specify whether ID3v2.4 or ID3v2.3 should be used.
*/
bool save(TagTypes tags, StripTags strip = StripOthers, ID3v2::Version version = ID3v2::v4);
/*! /*!
* Implements the unified property interface -- import function. * This will strip the tags that match the OR-ed together TagTypes from the file.
* This method forwards to ID3v2::Tag::setProperties(). * By default it strips all tags. It returns true if the tags are successfully stripped.
*/ *
PropertyMap setProperties(const PropertyMap &); * \note This will update the file immediately.
*/
void strip(TagTypes tags = AllTags);
/*! /*!
* Returns the AIFF::Properties for this file. If no audio properties * Returns whether or not the file on disk actually has an ID3v2 tag.
* were read then this will return a null pointer. *
*/ * \see ID3v2Tag()
virtual Properties *audioProperties() const; */
bool hasID3v2Tag() const;
/*! /*!
* Save the file. If at least one tag -- ID3v1 or DIIN -- exists this * Returns whether or not the file on disk actually has the DSDIFF title and artist tags.
* will duplicate its content into the other tag. This returns true *
* if saving was successful. * \see DIINTag()
* */
* If neither exists or if both tags are empty, this will strip the tags bool hasDIINTag() const;
* from the file.
*
* This is the same as calling save(AllTags);
*
* If you would like more granular control over the content of the tags,
* with the concession of generality, use paramaterized save call below.
*
* \see save(int tags)
*/
virtual bool save();
/*! /*!
* Save the file. This will attempt to save all of the tag types that are * Returns whether or not the given \a stream can be opened as a DSDIFF file.
* specified by OR-ing together TagTypes values. The save() method above *
* uses AllTags. This returns true if saving was successful. * \note This method is designed to do a quick check. The result may not necessarily be correct.
* */
* This strips all tags not included in the mask, but does not modify them static bool isSupported(IOStream *stream);
* in memory, so later calls to save() which make use of these tags will
* remain valid. This also strips empty tags.
*/
bool save(int tags);
/*! protected:
* Returns whether or not the file on disk actually has an ID3v2 tag. enum Endianness {
* BigEndian,
* \see ID3v2Tag() LittleEndian
*/ };
bool hasID3v2Tag() const;
/*! File(FileName file, Endianness endianness);
* Returns whether or not the file on disk actually has the DSDIFF File(IOStream *stream, Endianness endianness);
* Title & Artist tag.
*
* \see DIINTag()
*/
bool hasDIINTag() const;
/*! private:
* Returns whether or not the given \a stream can be opened as a DSDIFF File(const File &);
* file. File &operator=(const File &);
*
* \note This method is designed to do a quick check. The result may
* not necessarily be correct.
*/
static bool isSupported(IOStream *stream);
protected: void removeRootChunk(const ByteVector &id);
enum Endianness { BigEndian, LittleEndian }; void removeRootChunk(unsigned int i);
void removeChildChunk(unsigned int i, unsigned int childChunkNum);
File(FileName file, Endianness endianness); /*!
File(IOStream *stream, Endianness endianness); * Sets the data for the the specified chunk at root level to \a data.
*
* \warning This will update the file immediately.
*/
void setRootChunkData(unsigned int i, const ByteVector &data);
private: /*!
File(const File &); * Sets the data for the root-level chunk \a name to \a data.
File &operator=(const File &); * If a root-level chunk with the given name already exists it will be overwritten, otherwise it will be created after the existing chunks.
*
* \warning This will update the file immediately.
*/
void setRootChunkData(const ByteVector &name, const ByteVector &data);
/*! /*!
* Sets the data for the the specified chunk at root level to \a data. * Sets the data for the the specified child chunk to \a data.
* *
* \warning This will update the file immediately. * If data is null, then remove the chunk
*/ *
void setRootChunkData(unsigned int i, const ByteVector &data); * \warning This will update the file immediately.
*/
void setChildChunkData(unsigned int i, const ByteVector &data, unsigned int childChunkNum);
/*! /*!
* Sets the data for the root-level chunk \a name to \a data. * Sets the data for the child chunk \a name to \a data.
* If a root-level chunk with the given name already exists * If a chunk with the given name already exists it will be overwritten, otherwise it will be created after the existing chunks inside child chunk.
* it will be overwritten, otherwise it will be *
* created after the existing chunks. * If data is null, then remove the chunks with \a name name
* *
* \warning This will update the file immediately. * \warning This will update the file immediately.
*/ */
void setRootChunkData(const ByteVector &name, const ByteVector &data); void setChildChunkData(const ByteVector &name, const ByteVector &data, unsigned int childChunkNum);
/*! void updateRootChunksStructure(unsigned int startingChunk);
* Sets the data for the the specified child chunk to \a data.
*
* If data is null, then remove the chunk
*
* \warning This will update the file immediately.
*/
void setChildChunkData(unsigned int i, const ByteVector &data,
unsigned int childChunkNum);
/*! void read(bool readProperties, AudioProperties::ReadStyle propertiesStyle);
* Sets the data for the child chunk \a name to \a data. If a chunk with void writeChunk(const ByteVector &name, const ByteVector &data, unsigned long long offset, unsigned long replace = 0, unsigned int leadingPadding = 0);
* the given name already exists it will be overwritten, otherwise it will
* be created after the existing chunks inside child chunk.
*
* If data is null, then remove the chunks with \a name name
*
* \warning This will update the file immediately.
*/
void setChildChunkData(const ByteVector &name, const ByteVector &data,
unsigned int childChunkNum);
void updateRootChunksStructure(unsigned int startingChunk); class FilePrivate;
FilePrivate *d;
void read(bool readProperties, Properties::ReadStyle propertiesStyle); };
void writeChunk(const ByteVector &name, const ByteVector &data, } // namespace DSDIFF
unsigned long long offset, unsigned long replace = 0, } // namespace TagLib
unsigned int leadingPadding = 0); } // namespace Strawberry_TagLib
class FilePrivate;
FilePrivate *d;
};
}
}
}
#endif #endif

View File

@@ -23,24 +23,21 @@
* http://www.mozilla.org/MPL/ * * http://www.mozilla.org/MPL/ *
***************************************************************************/ ***************************************************************************/
#include <tstring.h> #include "tstring.h"
#include <tdebug.h> #include "tdebug.h"
#include "dsdiffproperties.h" #include "dsdiffproperties.h"
using namespace Strawberry_TagLib::TagLib; using namespace Strawberry_TagLib::TagLib;
class DSDIFF::Properties::PropertiesPrivate class DSDIFF::AudioProperties::AudioPropertiesPrivate {
{ public:
public: explicit AudioPropertiesPrivate() : length(0),
PropertiesPrivate() : bitrate(0),
length(0), sampleRate(0),
bitrate(0), channels(0),
sampleRate(0), sampleWidth(0),
channels(0), sampleCount(0) {
sampleWidth(0),
sampleCount(0)
{
} }
int length; int length;
@@ -55,66 +52,45 @@ public:
// public members // public members
//////////////////////////////////////////////////////////////////////////////// ////////////////////////////////////////////////////////////////////////////////
DSDIFF::Properties::Properties(const unsigned int sampleRate, DSDIFF::AudioProperties::AudioProperties(const unsigned int sampleRate, const unsigned short channels, const unsigned long long samplesCount, const int bitrate, ReadStyle) : Strawberry_TagLib::TagLib::AudioProperties(), d(new AudioPropertiesPrivate) {
const unsigned short channels,
const unsigned long long samplesCount,
const int bitrate,
ReadStyle style) : AudioProperties(style)
{
d = new PropertiesPrivate;
d->channels = channels; d->channels = channels;
d->sampleCount = samplesCount; d->sampleCount = samplesCount;
d->sampleWidth = 1; d->sampleWidth = 1;
d->sampleRate = sampleRate; d->sampleRate = sampleRate;
d->bitrate = bitrate; d->bitrate = bitrate;
d->length = d->sampleRate > 0 d->length = d->sampleRate > 0 ? static_cast<int>((d->sampleCount * 1000.0) / d->sampleRate + 0.5) : 0;
? static_cast<int>((d->sampleCount * 1000.0) / d->sampleRate + 0.5)
: 0;
} }
DSDIFF::Properties::~Properties() DSDIFF::AudioProperties::~AudioProperties() {
{
delete d; delete d;
} }
int DSDIFF::Properties::length() const int DSDIFF::AudioProperties::lengthInSeconds() const {
{
return lengthInSeconds();
}
int DSDIFF::Properties::lengthInSeconds() const
{
return d->length / 1000; return d->length / 1000;
} }
int DSDIFF::Properties::lengthInMilliseconds() const int DSDIFF::AudioProperties::lengthInMilliseconds() const {
{
return d->length; return d->length;
} }
int DSDIFF::Properties::bitrate() const int DSDIFF::AudioProperties::bitrate() const {
{
return d->bitrate; return d->bitrate;
} }
int DSDIFF::Properties::sampleRate() const int DSDIFF::AudioProperties::sampleRate() const {
{
return d->sampleRate; return d->sampleRate;
} }
int DSDIFF::Properties::channels() const int DSDIFF::AudioProperties::channels() const {
{
return d->channels; return d->channels;
} }
int DSDIFF::Properties::bitsPerSample() const int DSDIFF::AudioProperties::bitsPerSample() const {
{
return d->sampleWidth; return d->sampleWidth;
} }
long long DSDIFF::Properties::sampleCount() const long long DSDIFF::AudioProperties::sampleCount() const {
{
return d->sampleCount; return d->sampleCount;
} }

View File

@@ -30,56 +30,49 @@
namespace Strawberry_TagLib { namespace Strawberry_TagLib {
namespace TagLib { namespace TagLib {
namespace DSDIFF {
namespace DSDIFF { class File;
class File; //! An implementation of audio property reading for DSDIFF
//! An implementation of audio property reading for DSDIFF /*!
* This reads the data from an DSDIFF stream found in the AudioProperties API.
*/
/*! class TAGLIB_EXPORT AudioProperties : public Strawberry_TagLib::TagLib::AudioProperties {
* This reads the data from an DSDIFF stream found in the AudioProperties public:
* API. /*!
*/ * Create an instance of DSDIFF::AudioProperties with the data read from the ByteVector \a data.
*/
explicit AudioProperties(const unsigned int sampleRate, const unsigned short channels, const unsigned long long samplesCount, const int bitrate, ReadStyle);
class TAGLIB_EXPORT Properties : public AudioProperties /*!
{ * Destroys this DSDIFF::AudioProperties instance.
public: */
/*! ~AudioProperties() override;
* Create an instance of DSDIFF::Properties with the data read from the
* ByteVector \a data.
*/
Properties(const unsigned int sampleRate, const unsigned short channels,
const unsigned long long samplesCount, const int bitrate,
ReadStyle style);
/*! // Reimplementations.
* Destroys this DSDIFF::Properties instance.
*/
virtual ~Properties();
// Reimplementations. int lengthInSeconds() const override;
int lengthInMilliseconds() const override;
int bitrate() const override;
int sampleRate() const override;
int channels() const override;
virtual int length() const; int bitsPerSample() const;
virtual int lengthInSeconds() const; long long sampleCount() const;
virtual int lengthInMilliseconds() const;
virtual int bitrate() const;
virtual int sampleRate() const;
virtual int channels() const;
int bitsPerSample() const; private:
long long sampleCount() const; AudioProperties(const AudioProperties &);
AudioProperties &operator=(const AudioProperties &);
private: class AudioPropertiesPrivate;
Properties(const Properties &); AudioPropertiesPrivate *d;
Properties &operator=(const Properties &); };
class PropertiesPrivate; } // namespace DSDIFF
PropertiesPrivate *d; } // namespace TagLib
}; } // namespace Strawberry_TagLib
}
}
}
#endif #endif

View File

@@ -23,12 +23,14 @@
* http://www.mozilla.org/MPL/ * * http://www.mozilla.org/MPL/ *
***************************************************************************/ ***************************************************************************/
#include <tbytevector.h> #include <memory>
#include <tdebug.h>
#include <id3v2tag.h> #include "tbytevector.h"
#include <tstringlist.h> #include "tdebug.h"
#include <tpropertymap.h> #include "id3v2tag.h"
#include <tagutils.h> #include "tstringlist.h"
#include "tpropertymap.h"
#include "tagutils.h"
#include "dsffile.h" #include "dsffile.h"
@@ -36,38 +38,28 @@ using namespace Strawberry_TagLib::TagLib;
// The DSF specification is located at http://dsd-guide.com/sites/default/files/white-papers/DSFFileFormatSpec_E.pdf // The DSF specification is located at http://dsd-guide.com/sites/default/files/white-papers/DSFFileFormatSpec_E.pdf
class DSF::File::FilePrivate class DSF::File::FilePrivate {
{ public:
public: FilePrivate() : fileSize(0),
FilePrivate() : metadataOffset(0) {}
fileSize(0),
metadataOffset(0),
properties(nullptr),
tag(nullptr)
{
}
~FilePrivate()
{
delete properties;
delete tag;
}
long long fileSize; long long fileSize;
long long metadataOffset; long long metadataOffset;
Properties *properties;
ID3v2::Tag *tag; std::unique_ptr<AudioProperties> properties;
std::unique_ptr<ID3v2::Tag> tag;
}; };
//////////////////////////////////////////////////////////////////////////////// ////////////////////////////////////////////////////////////////////////////////
// static members // static members
//////////////////////////////////////////////////////////////////////////////// ////////////////////////////////////////////////////////////////////////////////
bool DSF::File::isSupported(IOStream *stream) bool DSF::File::isSupported(IOStream *stream) {
{
// A DSF file has to start with "DSD " // A DSF file has to start with "DSD "
const ByteVector id = Utils::readHeader(stream, 4, false); const ByteVector id = Utils::readHeader(stream, 4, false);
return id.startsWith("DSD "); return id.startsWith("DSD ");
} }
//////////////////////////////////////////////////////////////////////////////// ////////////////////////////////////////////////////////////////////////////////
@@ -75,74 +67,66 @@ bool DSF::File::isSupported(IOStream *stream)
//////////////////////////////////////////////////////////////////////////////// ////////////////////////////////////////////////////////////////////////////////
DSF::File::File(FileName file, bool readProperties, DSF::File::File(FileName file, bool readProperties,
Properties::ReadStyle propertiesStyle) : AudioProperties::ReadStyle propertiesStyle) : Strawberry_TagLib::TagLib::File(file), d(new FilePrivate()) {
Strawberry_TagLib::TagLib::File(file),
d(new FilePrivate()) if (isOpen())
{
if(isOpen())
read(readProperties, propertiesStyle); read(readProperties, propertiesStyle);
} }
DSF::File::File(IOStream *stream, bool readProperties, DSF::File::File(IOStream *stream, bool readProperties, AudioProperties::ReadStyle propertiesStyle) : Strawberry_TagLib::TagLib::File(stream), d(new FilePrivate()) {
Properties::ReadStyle propertiesStyle) :
Strawberry_TagLib::TagLib::File(stream), if (isOpen())
d(new FilePrivate())
{
if(isOpen())
read(readProperties, propertiesStyle); read(readProperties, propertiesStyle);
} }
DSF::File::~File() DSF::File::~File() {
{
delete d; delete d;
} }
ID3v2::Tag *DSF::File::tag() const ID3v2::Tag *DSF::File::tag() const {
{ return d->tag.get();
return d->tag;
} }
PropertyMap DSF::File::properties() const DSF::AudioProperties *DSF::File::audioProperties() const {
{ return d->properties.get();
}
PropertyMap DSF::File::properties() const {
return d->tag->properties(); return d->tag->properties();
} }
PropertyMap DSF::File::setProperties(const PropertyMap &properties) PropertyMap DSF::File::setProperties(const PropertyMap &properties) {
{
return d->tag->setProperties(properties); return d->tag->setProperties(properties);
} }
DSF::Properties *DSF::File::audioProperties() const bool DSF::File::save() {
{
return d->properties;
}
bool DSF::File::save() if (readOnly()) {
{
if(readOnly()) {
debug("DSF::File::save() -- File is read only."); debug("DSF::File::save() -- File is read only.");
return false; return false;
} }
if(!isValid()) { if (!isValid()) {
debug("DSF::File::save() -- Trying to save invalid file."); debug("DSF::File::save() -- Trying to save invalid file.");
return false; return false;
} }
// Three things must be updated: the file size, the tag data, and the metadata offset // Three things must be updated: the file size, the tag data, and the metadata offset
if(d->tag->isEmpty()) { if (d->tag->isEmpty()) {
long long newFileSize = d->metadataOffset ? d->metadataOffset : d->fileSize; long long newFileSize = d->metadataOffset ? d->metadataOffset : d->fileSize;
// Update the file size // Update the file size
if(d->fileSize != newFileSize) { if (d->fileSize != newFileSize) {
insert(ByteVector::fromLongLong(newFileSize, false), 12, 8); insert(ByteVector::fromUInt64LE(newFileSize), 12, 8);
d->fileSize = newFileSize; d->fileSize = newFileSize;
} }
// Update the metadata offset to 0 since there is no longer a tag // Update the metadata offset to 0 since there is no longer a tag
if(d->metadataOffset) { if (d->metadataOffset) {
insert(ByteVector::fromLongLong(0ULL, false), 20, 8); insert(ByteVector::fromUInt64LE(0ULL), 20, 8);
d->metadataOffset = 0; d->metadataOffset = 0;
} }
@@ -157,14 +141,14 @@ bool DSF::File::save()
long long oldTagSize = d->fileSize - newMetadataOffset; long long oldTagSize = d->fileSize - newMetadataOffset;
// Update the file size // Update the file size
if(d->fileSize != newFileSize) { if (d->fileSize != newFileSize) {
insert(ByteVector::fromLongLong(newFileSize, false), 12, 8); insert(ByteVector::fromUInt64LE(newFileSize), 12, 8);
d->fileSize = newFileSize; d->fileSize = newFileSize;
} }
// Update the metadata offset // Update the metadata offset
if(d->metadataOffset != newMetadataOffset) { if (d->metadataOffset != newMetadataOffset) {
insert(ByteVector::fromLongLong(newMetadataOffset, false), 20, 8); insert(ByteVector::fromUInt64LE(newMetadataOffset), 20, 8);
d->metadataOffset = newMetadataOffset; d->metadataOffset = newMetadataOffset;
} }
@@ -173,6 +157,7 @@ bool DSF::File::save()
} }
return true; return true;
} }
//////////////////////////////////////////////////////////////////////////////// ////////////////////////////////////////////////////////////////////////////////
@@ -180,41 +165,41 @@ bool DSF::File::save()
//////////////////////////////////////////////////////////////////////////////// ////////////////////////////////////////////////////////////////////////////////
void DSF::File::read(bool readProperties, Properties::ReadStyle propertiesStyle) void DSF::File::read(bool, AudioProperties::ReadStyle propertiesStyle) {
{
// A DSF file consists of four chunks: DSD chunk, format chunk, data chunk, and metadata chunk // A DSF file consists of four chunks: DSD chunk, format chunk, data chunk, and metadata chunk
// The file format is not chunked in the sense of a RIFF File, though // The file format is not chunked in the sense of a RIFF File, though
// DSD chunk // DSD chunk
ByteVector chunkName = readBlock(4); ByteVector chunkName = readBlock(4);
if(chunkName != "DSD ") { if (chunkName != "DSD ") {
debug("DSF::File::read() -- Not a DSF file."); debug("DSF::File::read() -- Not a DSF file.");
setValid(false); setValid(false);
return; return;
} }
long long chunkSize = readBlock(8).toLongLong(false); long long chunkSize = readBlock(8).toInt64LE(0);
// Integrity check // Integrity check
if(28 != chunkSize) { if (28 != chunkSize) {
debug("DSF::File::read() -- File is corrupted, wrong chunk size"); debug("DSF::File::read() -- File is corrupted, wrong chunk size");
setValid(false); setValid(false);
return; return;
} }
d->fileSize = readBlock(8).toLongLong(false); d->fileSize = readBlock(8).toInt64LE(0);
// File is malformed or corrupted // File is malformed or corrupted
if(d->fileSize != length()) { if (d->fileSize != length()) {
debug("DSF::File::read() -- File is corrupted wrong length"); debug("DSF::File::read() -- File is corrupted wrong length");
setValid(false); setValid(false);
return; return;
} }
d->metadataOffset = readBlock(8).toLongLong(false); d->metadataOffset = readBlock(8).toInt64LE(0);
// File is malformed or corrupted // File is malformed or corrupted
if(d->metadataOffset > d->fileSize) { if (d->metadataOffset > d->fileSize) {
debug("DSF::File::read() -- Invalid metadata offset."); debug("DSF::File::read() -- Invalid metadata offset.");
setValid(false); setValid(false);
return; return;
@@ -222,22 +207,22 @@ void DSF::File::read(bool readProperties, Properties::ReadStyle propertiesStyle)
// Format chunk // Format chunk
chunkName = readBlock(4); chunkName = readBlock(4);
if(chunkName != "fmt ") { if (chunkName != "fmt ") {
debug("DSF::File::read() -- Missing 'fmt ' chunk."); debug("DSF::File::read() -- Missing 'fmt ' chunk.");
setValid(false); setValid(false);
return; return;
} }
chunkSize = readBlock(8).toLongLong(false); chunkSize = readBlock(8).toInt64LE(0);
d->properties = new Properties(readBlock(chunkSize), propertiesStyle); d->properties.reset(new AudioProperties(readBlock(chunkSize), propertiesStyle));
// Skip the data chunk // Skip the data chunk
// A metadata offset of 0 indicates the absence of an ID3v2 tag // A metadata offset of 0 indicates the absence of an ID3v2 tag
if(0 == d->metadataOffset) if (0 == d->metadataOffset)
d->tag = new ID3v2::Tag(); d->tag.reset(new ID3v2::Tag());
else else
d->tag = new ID3v2::Tag(this, d->metadataOffset); d->tag.reset(new ID3v2::Tag(this, d->metadataOffset));
}
}

View File

@@ -33,98 +33,93 @@
namespace Strawberry_TagLib { namespace Strawberry_TagLib {
namespace TagLib { namespace TagLib {
//! An implementation of DSF metadata //! An implementation of DSF metadata
/*!
* This is implementation of DSF metadata.
*
* This supports an ID3v2 tag as well as properties from the file.
*/
namespace DSF {
//! An implementation of Strawberry_TagLib::TagLib::File with DSF specific methods
/*!
* This implements and provides an interface for DSF files to the
* Strawberry_TagLib::TagLib::Tag and Strawberry_TagLib::TagLib::AudioProperties interfaces by way of implementing
* the abstract Strawberry_TagLib::TagLib::File API as well as providing some additional information specific to DSF files.
*
*/
class TAGLIB_EXPORT File : public Strawberry_TagLib::TagLib::File {
public:
/*!
* Constructs an DSF file from \a file.
* If \a readProperties is true the file's audio properties will also be read using \a propertiesStyle.
* If false, \a propertiesStyle is ignored.
*/
explicit File(FileName file, bool readProperties = true, AudioProperties::ReadStyle propertiesStyle = AudioProperties::Average);
/*! /*!
* This is implementation of DSF metadata. * Constructs an DSF file from \a file.
* * If \a readProperties is true the file's audio properties will also be read using \a propertiesStyle.
* This supports an ID3v2 tag as well as properties from the file. * If false, \a propertiesStyle is ignored.
*/ */
explicit File(IOStream *stream, bool readProperties = true, AudioProperties::ReadStyle propertiesStyle = AudioProperties::Average);
namespace DSF { /*!
* Destroys this instance of the File.
*/
~File() override;
//! An implementation of Strawberry_TagLib::TagLib::File with DSF specific methods /*!
* Returns the Tag for this file.
*/
ID3v2::Tag *tag() const override;
/*! /*!
* This implements and provides an interface for DSF files to the * Implements the unified property interface -- export function.
* Strawberry_TagLib::TagLib::Tag and Strawberry_TagLib::TagLib::AudioProperties interfaces by way of implementing * This method forwards to ID3v2::Tag::properties().
* the abstract Strawberry_TagLib::TagLib::File API as well as providing some additional */
* information specific to DSF files. PropertyMap properties() const override;
*/
class TAGLIB_EXPORT File : public Strawberry_TagLib::TagLib::File /*!
{ * Implements the unified property interface -- import function.
public: * This method forwards to ID3v2::Tag::setProperties().
/*! */
* Contructs an DSF file from \a file. If \a readProperties is true the PropertyMap setProperties(const PropertyMap &) override;
* file's audio properties will also be read using \a propertiesStyle. If
* false, \a propertiesStyle is ignored.
*/
File(FileName file, bool readProperties = true,
Properties::ReadStyle propertiesStyle = Properties::Average);
/*! /*!
* Contructs an DSF file from \a file. If \a readProperties is true the * Returns the DSF::AudioProperties for this file.
* file's audio properties will also be read using \a propertiesStyle. If * If no audio properties were read then this will return a null pointer.
* false, \a propertiesStyle is ignored. */
*/ AudioProperties *audioProperties() const override;
File(IOStream *stream, bool readProperties = true,
Properties::ReadStyle propertiesStyle = Properties::Average);
/*! /*!
* Destroys this instance of the File. * Saves the file.
*/ */
virtual ~File(); bool save() override;
/*! /*!
* Returns the Tag for this file. * Returns whether or not the given \a stream can be opened as a DSF file.
*/ *
ID3v2::Tag *tag() const; * \note This method is designed to do a quick check.
* The result may not necessarily be correct.
*/
static bool isSupported(IOStream *stream);
/*! private:
* Implements the unified property interface -- export function. File(const File &);
* This method forwards to ID3v2::Tag::properties(). File &operator=(const File &);
*/
PropertyMap properties() const;
/*! void read(bool readProperties, AudioProperties::ReadStyle propertiesStyle);
* Implements the unified property interface -- import function.
* This method forwards to ID3v2::Tag::setProperties().
*/
PropertyMap setProperties(const PropertyMap &);
/*! class FilePrivate;
* Returns the DSF::AudioProperties for this file. If no audio properties FilePrivate *d;
* were read then this will return a null pointer. };
*/ } // namespace DSF
virtual Properties *audioProperties() const; } // namespace TagLib
} // namespace Strawberry_TagLib
/*!
* Saves the file.
*/
virtual bool save();
/*!
* Returns whether or not the given \a stream can be opened as a DSF
* file.
*
* \note This method is designed to do a quick check. The result may
* not necessarily be correct.
*/
static bool isSupported(IOStream *stream);
private:
File(const File &);
File &operator=(const File &);
void read(bool readProperties, Properties::ReadStyle propertiesStyle);
class FilePrivate;
FilePrivate *d;
};
}
}
}
#endif #endif

View File

@@ -23,28 +23,25 @@
* http://www.mozilla.org/MPL/ * * http://www.mozilla.org/MPL/ *
***************************************************************************/ ***************************************************************************/
#include <tstring.h> #include "tstring.h"
#include <tdebug.h> #include "tdebug.h"
#include "dsfproperties.h" #include "dsfproperties.h"
using namespace Strawberry_TagLib::TagLib; using namespace Strawberry_TagLib::TagLib;
class DSF::Properties::PropertiesPrivate class DSF::AudioProperties::AudioPropertiesPrivate {
{ public:
public: explicit AudioPropertiesPrivate() : formatVersion(0),
PropertiesPrivate() : formatID(0),
formatVersion(0), channelType(0),
formatID(0), channelNum(0),
channelType(0), samplingFrequency(0),
channelNum(0), bitsPerSample(0),
samplingFrequency(0), sampleCount(0),
bitsPerSample(0), blockSizePerChannel(0),
sampleCount(0), bitrate(0),
blockSizePerChannel(0), length(0) {
bitrate(0),
length(0)
{
} }
// Nomenclature is from DSF file format specification // Nomenclature is from DSF file format specification
@@ -58,83 +55,64 @@ public:
unsigned int blockSizePerChannel; unsigned int blockSizePerChannel;
// Computed // Computed
unsigned int bitrate; int bitrate;
unsigned int length; int length;
}; };
//////////////////////////////////////////////////////////////////////////////// ////////////////////////////////////////////////////////////////////////////////
// public members // public members
//////////////////////////////////////////////////////////////////////////////// ////////////////////////////////////////////////////////////////////////////////
DSF::Properties::Properties(const ByteVector &data, ReadStyle style) : Strawberry_TagLib::TagLib::AudioProperties(style) DSF::AudioProperties::AudioProperties(const ByteVector &data, ReadStyle) : Strawberry_TagLib::TagLib::AudioProperties(), d(new AudioPropertiesPrivate()) {
{
d = new PropertiesPrivate;
read(data); read(data);
} }
DSF::Properties::~Properties() DSF::AudioProperties::~AudioProperties() {
{
delete d; delete d;
} }
int DSF::Properties::length() const int DSF::AudioProperties::lengthInSeconds() const {
{
return lengthInSeconds();
}
int DSF::Properties::lengthInSeconds() const
{
return d->length / 1000; return d->length / 1000;
} }
int DSF::Properties::lengthInMilliseconds() const int DSF::AudioProperties::lengthInMilliseconds() const {
{
return d->length; return d->length;
} }
int DSF::Properties::bitrate() const int DSF::AudioProperties::bitrate() const {
{
return d->bitrate; return d->bitrate;
} }
int DSF::Properties::sampleRate() const int DSF::AudioProperties::sampleRate() const {
{
return d->samplingFrequency; return d->samplingFrequency;
} }
int DSF::Properties::channels() const int DSF::AudioProperties::channels() const {
{
return d->channelNum; return d->channelNum;
} }
// DSF specific // DSF specific
int DSF::Properties::formatVersion() const int DSF::AudioProperties::formatVersion() const {
{
return d->formatVersion; return d->formatVersion;
} }
int DSF::Properties::formatID() const int DSF::AudioProperties::formatID() const {
{
return d->formatID; return d->formatID;
} }
int DSF::Properties::channelType() const int DSF::AudioProperties::channelType() const {
{
return d->channelType; return d->channelType;
} }
int DSF::Properties::bitsPerSample() const int DSF::AudioProperties::bitsPerSample() const {
{
return d->bitsPerSample; return d->bitsPerSample;
} }
long long DSF::Properties::sampleCount() const long long DSF::AudioProperties::sampleCount() const {
{
return d->sampleCount; return d->sampleCount;
} }
int DSF::Properties::blockSizePerChannel() const int DSF::AudioProperties::blockSizePerChannel() const {
{
return d->blockSizePerChannel; return d->blockSizePerChannel;
} }
@@ -142,20 +120,16 @@ int DSF::Properties::blockSizePerChannel() const
// private members // private members
//////////////////////////////////////////////////////////////////////////////// ////////////////////////////////////////////////////////////////////////////////
void DSF::Properties::read(const ByteVector &data) void DSF::AudioProperties::read(const ByteVector &data) {
{ d->formatVersion = data.toUInt32LE(0);
d->formatVersion = data.toUInt(0U,false); d->formatID = data.toUInt32LE(4);
d->formatID = data.toUInt(4U,false); d->channelType = data.toUInt32LE(8);
d->channelType = data.toUInt(8U,false); d->channelNum = data.toUInt32LE(12);
d->channelNum = data.toUInt(12U,false); d->samplingFrequency = data.toUInt32LE(16);
d->samplingFrequency = data.toUInt(16U,false); d->bitsPerSample = data.toUInt32LE(20);
d->bitsPerSample = data.toUInt(20U,false); d->sampleCount = data.toInt64LE(24);
d->sampleCount = data.toLongLong(24U,false); d->blockSizePerChannel = data.toUInt32LE(32);
d->blockSizePerChannel = data.toUInt(32U,false);
d->bitrate d->bitrate = static_cast<int>((d->samplingFrequency * d->bitsPerSample * d->channelNum) / 1000.0 + 0.5);
= static_cast<unsigned int>((d->samplingFrequency * d->bitsPerSample * d->channelNum) / 1000.0 + 0.5); d->length = d->samplingFrequency > 0 ? static_cast<int>(d->sampleCount * 1000.0 / d->samplingFrequency + 0.5) : 0;
d->length
= d->samplingFrequency > 0 ? static_cast<unsigned int>(d->sampleCount * 1000.0 / d->samplingFrequency + 0.5) : 0;
} }

View File

@@ -30,65 +30,65 @@
namespace Strawberry_TagLib { namespace Strawberry_TagLib {
namespace TagLib { namespace TagLib {
namespace DSF {
namespace DSF { class File;
class File; //! An implementation of audio property reading for DSF
//! An implementation of audio property reading for DSF /*!
* This reads the data from a DSF stream found in the AudioProperties API.
*/
/*! class TAGLIB_EXPORT AudioProperties : public Strawberry_TagLib::TagLib::AudioProperties {
* This reads the data from a DSF stream found in the AudioProperties public:
* API. /*!
*/ * Create an instance of DSF::AudioProperties with the data read from the ByteVector \a data.
*/
explicit AudioProperties(const ByteVector &data, ReadStyle);
class TAGLIB_EXPORT Properties : public Strawberry_TagLib::TagLib::AudioProperties /*!
{ * Destroys this DSF::AudioProperties instance.
public: */
/*! ~AudioProperties() override;
* Create an instance of DSF::AudioProperties with the data read from the
* ByteVector \a data.
*/
Properties(const ByteVector &data, ReadStyle style);
/*! // Reimplementations.
* Destroys this DSF::AudioProperties instance.
*/
virtual ~Properties();
// Reimplementations. int lengthInSeconds() const override;
int lengthInMilliseconds() const override;
int bitrate() const override;
int sampleRate() const override;
int channels() const override;
virtual int length() const; int formatVersion() const;
virtual int lengthInSeconds() const; int formatID() const;
virtual int lengthInMilliseconds() const;
virtual int bitrate() const;
virtual int sampleRate() const;
virtual int channels() const;
int formatVersion() const; /*!
int formatID() const; * Channel type values:
* 1 = mono,
* 2 = stereo,
* 3 = 3 channels,
* 4 = quad,
* 5 = 4 channels,
* 6 = 5 channels,
* 7 = 5.1 channels
*/
int channelType() const;
int bitsPerSample() const;
long long sampleCount() const;
int blockSizePerChannel() const;
/*! private:
* Channel type values: 1 = mono, 2 = stereo, 3 = 3 channels, AudioProperties(const AudioProperties&);
* 4 = quad, 5 = 4 channels, 6 = 5 channels, 7 = 5.1 channels AudioProperties &operator=(const AudioProperties&);
*/
int channelType() const;
int bitsPerSample() const;
long long sampleCount() const;
int blockSizePerChannel() const;
private: void read(const ByteVector &data);
Properties(const Properties &);
Properties &operator=(const Properties &);
void read(const ByteVector &data); class AudioPropertiesPrivate;
AudioPropertiesPrivate *d;
class PropertiesPrivate; };
PropertiesPrivate *d; } // namespace DSF
}; } // namespace TagLib
} } // namespace Strawberry_TagLib
}
}
#endif #endif

View File

@@ -27,11 +27,12 @@
* http://www.mozilla.org/MPL/ * * http://www.mozilla.org/MPL/ *
***************************************************************************/ ***************************************************************************/
#include <tfile.h> #include <memory>
#include <tfilestream.h>
#include <tstring.h> #include "tfile.h"
#include <tdebug.h> #include "tfilestream.h"
#include <trefcounter.h> #include "tstring.h"
#include "tdebug.h"
#include "fileref.h" #include "fileref.h"
#include "asffile.h" #include "asffile.h"
@@ -57,320 +58,259 @@
using namespace Strawberry_TagLib::TagLib; using namespace Strawberry_TagLib::TagLib;
namespace namespace {
{ typedef List<const FileRef::FileTypeResolver*> ResolverList;
typedef List<const FileRef::FileTypeResolver *> ResolverList; ResolverList fileTypeResolvers;
ResolverList fileTypeResolvers;
// Detect the file type by user-defined resolvers. // Detect the file type by user-defined resolvers.
File *detectByResolvers(FileName fileName, bool readAudioProperties, File *detectByResolvers(FileName fileName, bool readAudioProperties, AudioProperties::ReadStyle audioPropertiesStyle) {
AudioProperties::ReadStyle audioPropertiesStyle)
{
ResolverList::ConstIterator it = fileTypeResolvers.begin();
for(; it != fileTypeResolvers.end(); ++it) {
File *file = (*it)->createFile(fileName, readAudioProperties, audioPropertiesStyle);
if(file)
return file;
}
return nullptr; ResolverList::ConstIterator it = fileTypeResolvers.begin();
} for (; it != fileTypeResolvers.end(); ++it) {
File *file = (*it)->createFile(fileName, readAudioProperties, audioPropertiesStyle);
// Detect the file type based on the file extension. if (file)
File* detectByExtension(IOStream *stream, bool readAudioProperties,
AudioProperties::ReadStyle audioPropertiesStyle)
{
#ifdef _WIN32
const String s = stream->name().toString();
#else
const String s(stream->name());
#endif
String ext;
const int pos = s.rfind(".");
if(pos != -1)
ext = s.substr(pos + 1).upper();
// If this list is updated, the method defaultFileExtensions() should also be
// updated. However at some point that list should be created at the same time
// that a default file type resolver is created.
if(ext.isEmpty())
return nullptr;
// .oga can be any audio in the Ogg container. So leave it to content-based detection.
if(ext == "MP3")
return new MPEG::File(stream, ID3v2::FrameFactory::instance(), readAudioProperties, audioPropertiesStyle);
if(ext == "OGG")
return new Ogg::Vorbis::File(stream, readAudioProperties, audioPropertiesStyle);
if(ext == "FLAC")
return new FLAC::File(stream, ID3v2::FrameFactory::instance(), readAudioProperties, audioPropertiesStyle);
if(ext == "MPC")
return new MPC::File(stream, readAudioProperties, audioPropertiesStyle);
if(ext == "WV")
return new WavPack::File(stream, readAudioProperties, audioPropertiesStyle);
if(ext == "SPX")
return new Ogg::Speex::File(stream, readAudioProperties, audioPropertiesStyle);
if(ext == "OPUS")
return new Ogg::Opus::File(stream, readAudioProperties, audioPropertiesStyle);
if(ext == "TTA")
return new TrueAudio::File(stream, readAudioProperties, audioPropertiesStyle);
if(ext == "M4A" || ext == "M4R" || ext == "M4B" || ext == "M4P" || ext == "MP4" || ext == "3G2" || ext == "M4V")
return new MP4::File(stream, readAudioProperties, audioPropertiesStyle);
if(ext == "WMA" || ext == "ASF")
return new ASF::File(stream, readAudioProperties, audioPropertiesStyle);
if(ext == "AIF" || ext == "AIFF" || ext == "AFC" || ext == "AIFC")
return new RIFF::AIFF::File(stream, readAudioProperties, audioPropertiesStyle);
if(ext == "WAV")
return new RIFF::WAV::File(stream, readAudioProperties, audioPropertiesStyle);
if(ext == "APE")
return new APE::File(stream, readAudioProperties, audioPropertiesStyle);
// module, nst and wow are possible but uncommon extensions
if(ext == "MOD" || ext == "MODULE" || ext == "NST" || ext == "WOW")
return new Mod::File(stream, readAudioProperties, audioPropertiesStyle);
if(ext == "S3M")
return new S3M::File(stream, readAudioProperties, audioPropertiesStyle);
if(ext == "IT")
return new IT::File(stream, readAudioProperties, audioPropertiesStyle);
if(ext == "XM")
return new XM::File(stream, readAudioProperties, audioPropertiesStyle);
if(ext == "DFF" || ext == "DSDIFF")
return new DSDIFF::File(stream, readAudioProperties, audioPropertiesStyle);
if(ext == "DSF")
return new DSF::File(stream, readAudioProperties, audioPropertiesStyle);
return nullptr;
}
// Detect the file type based on the actual content of the stream.
File *detectByContent(IOStream *stream, bool readAudioProperties,
AudioProperties::ReadStyle audioPropertiesStyle)
{
File *file = 0;
if(MPEG::File::isSupported(stream))
file = new MPEG::File(stream, ID3v2::FrameFactory::instance(), readAudioProperties, audioPropertiesStyle);
else if(Ogg::Vorbis::File::isSupported(stream))
file = new Ogg::Vorbis::File(stream, readAudioProperties, audioPropertiesStyle);
else if(Ogg::FLAC::File::isSupported(stream))
file = new Ogg::FLAC::File(stream, readAudioProperties, audioPropertiesStyle);
else if(FLAC::File::isSupported(stream))
file = new FLAC::File(stream, ID3v2::FrameFactory::instance(), readAudioProperties, audioPropertiesStyle);
else if(MPC::File::isSupported(stream))
file = new MPC::File(stream, readAudioProperties, audioPropertiesStyle);
else if(WavPack::File::isSupported(stream))
file = new WavPack::File(stream, readAudioProperties, audioPropertiesStyle);
else if(Ogg::Speex::File::isSupported(stream))
file = new Ogg::Speex::File(stream, readAudioProperties, audioPropertiesStyle);
else if(Ogg::Opus::File::isSupported(stream))
file = new Ogg::Opus::File(stream, readAudioProperties, audioPropertiesStyle);
else if(TrueAudio::File::isSupported(stream))
file = new TrueAudio::File(stream, readAudioProperties, audioPropertiesStyle);
else if(MP4::File::isSupported(stream))
file = new MP4::File(stream, readAudioProperties, audioPropertiesStyle);
else if(ASF::File::isSupported(stream))
file = new ASF::File(stream, readAudioProperties, audioPropertiesStyle);
else if(RIFF::AIFF::File::isSupported(stream))
file = new RIFF::AIFF::File(stream, readAudioProperties, audioPropertiesStyle);
else if(RIFF::WAV::File::isSupported(stream))
file = new RIFF::WAV::File(stream, readAudioProperties, audioPropertiesStyle);
else if(APE::File::isSupported(stream))
file = new APE::File(stream, readAudioProperties, audioPropertiesStyle);
else if(DSDIFF::File::isSupported(stream))
file = new DSDIFF::File(stream, readAudioProperties, audioPropertiesStyle);
else if(DSF::File::isSupported(stream))
file = new DSF::File(stream, readAudioProperties, audioPropertiesStyle);
// isSupported() only does a quick check, so double check the file here.
if(file) {
if(file->isValid())
return file;
else
delete file;
}
return nullptr;
}
// Internal function that supports FileRef::create().
// This looks redundant, but necessary in order not to change the previous
// behavior of FileRef::create().
File* createInternal(FileName fileName, bool readAudioProperties,
AudioProperties::ReadStyle audioPropertiesStyle)
{
File *file = detectByResolvers(fileName, readAudioProperties, audioPropertiesStyle);
if(file)
return file; return file;
#ifdef _WIN32
const String s = fileName.toString();
#else
const String s(fileName);
#endif
String ext;
const int pos = s.rfind(".");
if(pos != -1)
ext = s.substr(pos + 1).upper();
if(ext.isEmpty())
return nullptr;
if(ext == "MP3")
return new MPEG::File(fileName, ID3v2::FrameFactory::instance(), readAudioProperties, audioPropertiesStyle);
if(ext == "OGG")
return new Ogg::Vorbis::File(fileName, readAudioProperties, audioPropertiesStyle);
if(ext == "OGA") {
/* .oga can be any audio in the Ogg container. First try FLAC, then Vorbis. */
File *file = new Ogg::FLAC::File(fileName, readAudioProperties, audioPropertiesStyle);
if(file->isValid())
return file;
delete file;
return new Ogg::Vorbis::File(fileName, readAudioProperties, audioPropertiesStyle);
}
if(ext == "FLAC")
return new FLAC::File(fileName, ID3v2::FrameFactory::instance(), readAudioProperties, audioPropertiesStyle);
if(ext == "MPC")
return new MPC::File(fileName, readAudioProperties, audioPropertiesStyle);
if(ext == "WV")
return new WavPack::File(fileName, readAudioProperties, audioPropertiesStyle);
if(ext == "SPX")
return new Ogg::Speex::File(fileName, readAudioProperties, audioPropertiesStyle);
if(ext == "OPUS")
return new Ogg::Opus::File(fileName, readAudioProperties, audioPropertiesStyle);
if(ext == "TTA")
return new TrueAudio::File(fileName, readAudioProperties, audioPropertiesStyle);
if(ext == "M4A" || ext == "M4R" || ext == "M4B" || ext == "M4P" || ext == "MP4" || ext == "3G2" || ext == "M4V")
return new MP4::File(fileName, readAudioProperties, audioPropertiesStyle);
if(ext == "WMA" || ext == "ASF")
return new ASF::File(fileName, readAudioProperties, audioPropertiesStyle);
if(ext == "AIF" || ext == "AIFF" || ext == "AFC" || ext == "AIFC")
return new RIFF::AIFF::File(fileName, readAudioProperties, audioPropertiesStyle);
if(ext == "WAV")
return new RIFF::WAV::File(fileName, readAudioProperties, audioPropertiesStyle);
if(ext == "APE")
return new APE::File(fileName, readAudioProperties, audioPropertiesStyle);
// module, nst and wow are possible but uncommon extensions
if(ext == "MOD" || ext == "MODULE" || ext == "NST" || ext == "WOW")
return new Mod::File(fileName, readAudioProperties, audioPropertiesStyle);
if(ext == "S3M")
return new S3M::File(fileName, readAudioProperties, audioPropertiesStyle);
if(ext == "IT")
return new IT::File(fileName, readAudioProperties, audioPropertiesStyle);
if(ext == "XM")
return new XM::File(fileName, readAudioProperties, audioPropertiesStyle);
if(ext == "DFF" || ext == "DSDIFF")
return new DSDIFF::File(fileName, readAudioProperties, audioPropertiesStyle);
if(ext == "DSF")
return new DSF::File(fileName, readAudioProperties, audioPropertiesStyle);
return nullptr;
} }
return nullptr;
} }
class FileRef::FileRefPrivate : public RefCounter // Detect the file type based on the file extension.
{
public:
FileRefPrivate() :
RefCounter(),
file(0),
stream(0) {}
~FileRefPrivate() { File *detectByExtension(IOStream *stream, bool readAudioProperties, AudioProperties::ReadStyle audioPropertiesStyle) {
#ifdef _WIN32
const String s(stream->name().wstr());
#else
const String s(stream->name());
#endif
String ext;
const size_t pos = s.rfind(".");
if (pos != String::npos())
ext = s.substr(pos + 1).upper();
// If this list is updated, the method defaultFileExtensions() should also be
// updated. However at some point that list should be created at the same time
// that a default file type resolver is created.
if (ext.isEmpty())
return nullptr;
// .oga can be any audio in the Ogg container. So leave it to content-based detection.
if (ext == "MP3")
return new MPEG::File(stream, ID3v2::FrameFactory::instance(), readAudioProperties, audioPropertiesStyle);
if (ext == "OGG")
return new Ogg::Vorbis::File(stream, readAudioProperties, audioPropertiesStyle);
if (ext == "FLAC")
return new FLAC::File(stream, readAudioProperties, audioPropertiesStyle);
if (ext == "MPC")
return new MPC::File(stream, readAudioProperties, audioPropertiesStyle);
if (ext == "WV")
return new WavPack::File(stream, readAudioProperties, audioPropertiesStyle);
if (ext == "SPX")
return new Ogg::Speex::File(stream, readAudioProperties, audioPropertiesStyle);
if (ext == "OPUS")
return new Ogg::Opus::File(stream, readAudioProperties, audioPropertiesStyle);
if (ext == "TTA")
return new TrueAudio::File(stream, readAudioProperties, audioPropertiesStyle);
if (ext == "M4A" || ext == "M4R" || ext == "M4B" || ext == "M4P" || ext == "MP4" || ext == "3G2" || ext == "M4V")
return new MP4::File(stream, readAudioProperties, audioPropertiesStyle);
if (ext == "WMA" || ext == "ASF")
return new ASF::File(stream, readAudioProperties, audioPropertiesStyle);
if (ext == "AIF" || ext == "AIFF" || ext == "AFC" || ext == "AIFC")
return new RIFF::AIFF::File(stream, readAudioProperties, audioPropertiesStyle);
if (ext == "WAV")
return new RIFF::WAV::File(stream, readAudioProperties, audioPropertiesStyle);
if (ext == "APE")
return new APE::File(stream, readAudioProperties, audioPropertiesStyle);
// module, nst and wow are possible but uncommon extensions
if (ext == "MOD" || ext == "MODULE" || ext == "NST" || ext == "WOW")
return new Mod::File(stream, readAudioProperties, audioPropertiesStyle);
if (ext == "S3M")
return new S3M::File(stream, readAudioProperties, audioPropertiesStyle);
if (ext == "IT")
return new IT::File(stream, readAudioProperties, audioPropertiesStyle);
if (ext == "XM")
return new XM::File(stream, readAudioProperties, audioPropertiesStyle);
if (ext == "DFF" || ext == "DSDIFF")
return new DSDIFF::File(stream, readAudioProperties, audioPropertiesStyle);
if (ext == "DSF")
return new DSF::File(stream, readAudioProperties, audioPropertiesStyle);
return nullptr;
}
// Detect the file type based on the actual content of the stream.
File *detectByContent(IOStream *stream, bool readAudioProperties, AudioProperties::ReadStyle audioPropertiesStyle) {
File *file = nullptr;
if (MPEG::File::isSupported(stream))
file = new MPEG::File(stream, ID3v2::FrameFactory::instance(), readAudioProperties, audioPropertiesStyle);
else if (Ogg::Vorbis::File::isSupported(stream))
file = new Ogg::Vorbis::File(stream, readAudioProperties, audioPropertiesStyle);
else if (Ogg::FLAC::File::isSupported(stream))
file = new Ogg::FLAC::File(stream, readAudioProperties, audioPropertiesStyle);
else if (FLAC::File::isSupported(stream))
file = new FLAC::File(stream, readAudioProperties, audioPropertiesStyle);
else if (MPC::File::isSupported(stream))
file = new MPC::File(stream, readAudioProperties, audioPropertiesStyle);
else if (WavPack::File::isSupported(stream))
file = new WavPack::File(stream, readAudioProperties, audioPropertiesStyle);
else if (Ogg::Speex::File::isSupported(stream))
file = new Ogg::Speex::File(stream, readAudioProperties, audioPropertiesStyle);
else if (Ogg::Opus::File::isSupported(stream))
file = new Ogg::Opus::File(stream, readAudioProperties, audioPropertiesStyle);
else if (TrueAudio::File::isSupported(stream))
file = new TrueAudio::File(stream, readAudioProperties, audioPropertiesStyle);
else if (MP4::File::isSupported(stream))
file = new MP4::File(stream, readAudioProperties, audioPropertiesStyle);
else if (ASF::File::isSupported(stream))
file = new ASF::File(stream, readAudioProperties, audioPropertiesStyle);
else if (RIFF::AIFF::File::isSupported(stream))
file = new RIFF::AIFF::File(stream, readAudioProperties, audioPropertiesStyle);
else if (RIFF::WAV::File::isSupported(stream))
file = new RIFF::WAV::File(stream, readAudioProperties, audioPropertiesStyle);
else if (APE::File::isSupported(stream))
file = new APE::File(stream, readAudioProperties, audioPropertiesStyle);
else if (DSDIFF::File::isSupported(stream))
file = new DSDIFF::File(stream, readAudioProperties, audioPropertiesStyle);
else if (DSF::File::isSupported(stream))
file = new DSF::File(stream, readAudioProperties, audioPropertiesStyle);
// isSupported() only does a quick check, so double check the file here.
if (file) {
if (file->isValid())
return file;
else
delete file;
}
return nullptr;
}
struct FileRefData {
FileRefData() : file(nullptr), stream(nullptr) {}
~FileRefData() {
delete file; delete file;
delete stream; delete stream;
} }
File *file; File *file;
IOStream *stream; IOStream *stream;
}; };
} // namespace
class FileRef::FileRefPrivate {
public:
FileRefPrivate() : data(new FileRefData()) {}
std::shared_ptr<FileRefData> data;
};
//////////////////////////////////////////////////////////////////////////////// ////////////////////////////////////////////////////////////////////////////////
// public members // public members
//////////////////////////////////////////////////////////////////////////////// ////////////////////////////////////////////////////////////////////////////////
FileRef::FileRef() : FileRef::FileRef() : d(new FileRefPrivate()) {}
d(new FileRefPrivate())
{
}
FileRef::FileRef(FileName fileName, bool readAudioProperties, FileRef::FileRef(FileName fileName, bool readAudioProperties, AudioProperties::ReadStyle audioPropertiesStyle) : d(new FileRefPrivate()) {
AudioProperties::ReadStyle audioPropertiesStyle) :
d(new FileRefPrivate())
{
parse(fileName, readAudioProperties, audioPropertiesStyle); parse(fileName, readAudioProperties, audioPropertiesStyle);
} }
FileRef::FileRef(IOStream* stream, bool readAudioProperties, AudioProperties::ReadStyle audioPropertiesStyle) : FileRef::FileRef(IOStream *stream, bool readAudioProperties, AudioProperties::ReadStyle audioPropertiesStyle) : d(new FileRefPrivate()) {
d(new FileRefPrivate())
{
parse(stream, readAudioProperties, audioPropertiesStyle); parse(stream, readAudioProperties, audioPropertiesStyle);
} }
FileRef::FileRef(File *file) : FileRef::FileRef(File *file) : d(new FileRefPrivate()) {
d(new FileRefPrivate()) d->data->file = file;
{
d->file = file;
} }
FileRef::FileRef(const FileRef &ref) : FileRef::FileRef(const FileRef &ref) : d(new FileRefPrivate(*ref.d)) {}
d(ref.d)
{ FileRef::~FileRef() {
d->ref(); delete d;
} }
FileRef::~FileRef() Tag *FileRef::tag() const {
{
if(d->deref())
delete d;
}
Tag *FileRef::tag() const if (isNull()) {
{
if(isNull()) {
debug("FileRef::tag() - Called without a valid file."); debug("FileRef::tag() - Called without a valid file.");
return nullptr; return nullptr;
} }
return d->file->tag(); return d->data->file->tag();
} }
AudioProperties *FileRef::audioProperties() const PropertyMap FileRef::properties() const {
{ if (isNull()) {
if(isNull()) { debug("FileRef::properties() - Called without a valid file.");
return PropertyMap();
}
return d->data->file->properties();
}
void FileRef::removeUnsupportedProperties(const StringList &properties) {
if (isNull()) {
debug("FileRef::removeUnsupportedProperties() - Called without a valid file.");
return;
}
d->data->file->removeUnsupportedProperties(properties);
}
PropertyMap FileRef::setProperties(const PropertyMap &properties) {
if (isNull()) {
debug("FileRef::setProperties() - Called without a valid file.");
return PropertyMap();
}
return d->data->file->setProperties(properties);
}
AudioProperties *FileRef::audioProperties() const {
if (isNull()) {
debug("FileRef::audioProperties() - Called without a valid file."); debug("FileRef::audioProperties() - Called without a valid file.");
return nullptr; return nullptr;
} }
return d->file->audioProperties(); return d->data->file->audioProperties();
} }
File *FileRef::file() const File *FileRef::file() const {
{ return d->data->file;
return d->file;
} }
bool FileRef::save() bool FileRef::save() {
{
if(isNull()) { if (isNull()) {
debug("FileRef::save() - Called without a valid file."); debug("FileRef::save() - Called without a valid file.");
return false; return false;
} }
return d->file->save(); return d->data->file->save();
} }
const FileRef::FileTypeResolver *FileRef::addFileTypeResolver(const FileRef::FileTypeResolver *resolver) // static const FileRef::FileTypeResolver *FileRef::addFileTypeResolver(const FileRef::FileTypeResolver *resolver) { // static
{
fileTypeResolvers.prepend(resolver); fileTypeResolvers.prepend(resolver);
return resolver; return resolver;
} }
StringList FileRef::defaultFileExtensions() StringList FileRef::defaultFileExtensions() {
{
StringList l; StringList l;
l.append("ogg"); l.append("ogg");
@@ -395,97 +335,90 @@ StringList FileRef::defaultFileExtensions()
l.append("wav"); l.append("wav");
l.append("ape"); l.append("ape");
l.append("mod"); l.append("mod");
l.append("module"); // alias for "mod" l.append("module"); // alias for "mod"
l.append("nst"); // alias for "mod" l.append("nst"); // alias for "mod"
l.append("wow"); // alias for "mod" l.append("wow"); // alias for "mod"
l.append("s3m"); l.append("s3m");
l.append("it"); l.append("it");
l.append("xm"); l.append("xm");
l.append("dsf"); l.append("dsf");
l.append("dff"); l.append("dff");
l.append("dsdiff"); // alias for "dff" l.append("dsdiff"); // alias for "dff"
return l; return l;
} }
bool FileRef::isNull() const bool FileRef::isValid() const {
{ return (d->data->file && d->data->file->isValid());
return (!d->file || !d->file->isValid());
} }
FileRef &FileRef::operator=(const FileRef &ref) bool FileRef::isNull() const {
{ return (!d->data->file || !d->data->file->isValid());
}
FileRef &FileRef::operator=(const FileRef &ref) {
FileRef(ref).swap(*this); FileRef(ref).swap(*this);
return *this; return *this;
} }
void FileRef::swap(FileRef &ref) void FileRef::swap(FileRef &ref) {
{
using std::swap; using std::swap;
swap(d, ref.d); swap(d, ref.d);
} }
bool FileRef::operator==(const FileRef &ref) const bool FileRef::operator==(const FileRef &ref) const {
{ return (ref.d->data == d->data);
return (ref.d->file == d->file);
} }
bool FileRef::operator!=(const FileRef &ref) const bool FileRef::operator!=(const FileRef &ref) const {
{ return (ref.d->data != d->data);
return (ref.d->file != d->file);
}
File *FileRef::create(FileName fileName, bool readAudioProperties,
AudioProperties::ReadStyle audioPropertiesStyle) // static
{
return createInternal(fileName, readAudioProperties, audioPropertiesStyle);
} }
//////////////////////////////////////////////////////////////////////////////// ////////////////////////////////////////////////////////////////////////////////
// private members // private members
//////////////////////////////////////////////////////////////////////////////// ////////////////////////////////////////////////////////////////////////////////
void FileRef::parse(FileName fileName, bool readAudioProperties, void FileRef::parse(FileName fileName, bool readAudioProperties, AudioProperties::ReadStyle audioPropertiesStyle) {
AudioProperties::ReadStyle audioPropertiesStyle)
{
// Try user-defined resolvers. // Try user-defined resolvers.
d->file = detectByResolvers(fileName, readAudioProperties, audioPropertiesStyle); d->data->file = detectByResolvers(fileName, readAudioProperties, audioPropertiesStyle);
if(d->file) if (d->data->file)
return; return;
// Try to resolve file types based on the file extension. // Try to resolve file types based on the file extension.
d->stream = new FileStream(fileName); d->data->stream = new FileStream(fileName);
d->file = detectByExtension(d->stream, readAudioProperties, audioPropertiesStyle); d->data->file = detectByExtension(d->data->stream, readAudioProperties, audioPropertiesStyle);
if(d->file) if (d->data->file)
return; return;
// At last, try to resolve file types based on the actual content. // At last, try to resolve file types based on the actual content.
d->file = detectByContent(d->stream, readAudioProperties, audioPropertiesStyle); d->data->file = detectByContent(d->data->stream, readAudioProperties, audioPropertiesStyle);
if(d->file) if (d->data->file)
return; return;
// Stream have to be closed here if failed to resolve file types. // Stream have to be closed here if failed to resolve file types.
delete d->stream; delete d->data->stream;
d->stream = 0; d->data->stream = nullptr;
} }
void FileRef::parse(IOStream *stream, bool readAudioProperties, void FileRef::parse(IOStream *stream, bool readAudioProperties, AudioProperties::ReadStyle audioPropertiesStyle) {
AudioProperties::ReadStyle audioPropertiesStyle)
{
// User-defined resolvers won't work with a stream. // User-defined resolvers won't work with a stream.
// Try to resolve file types based on the file extension. // Try to resolve file types based on the file extension.
d->file = detectByExtension(stream, readAudioProperties, audioPropertiesStyle); d->data->file = detectByExtension(stream, readAudioProperties, audioPropertiesStyle);
if(d->file) if (d->data->file)
return; return;
// At last, try to resolve file types based on the actual content of the file. // At last, try to resolve file types based on the actual content of the file.
d->file = detectByContent(stream, readAudioProperties, audioPropertiesStyle); d->data->file = detectByContent(stream, readAudioProperties, audioPropertiesStyle);
} }

View File

@@ -30,42 +30,37 @@
#include "tstringlist.h" #include "tstringlist.h"
#include "taglib_export.h" #include "taglib_export.h"
#include "tpropertymap.h"
#include "audioproperties.h" #include "audioproperties.h"
namespace Strawberry_TagLib { namespace Strawberry_TagLib {
namespace TagLib { namespace TagLib {
class Tag; class Tag;
//! This class provides a simple abstraction for creating and handling files //! This class provides a simple abstraction for creating and handling files
/*! /*!
* FileRef exists to provide a minimal, generic and value-based wrapper around * FileRef exists to provide a minimal, generic and value-based wrapper around a File.
* a File. It is lightweight and implicitly shared, and as such suitable for * It is lightweight and implicitly shared, and as such suitable for pass-by-value use.
* pass-by-value use. This hides some of the uglier details of Strawberry_TagLib::TagLib::File * This hides some of the uglier details of TagLib::File and the non-generic portions of the concrete file implementations.
* and the non-generic portions of the concrete file implementations.
* *
* This class is useful in a "simple usage" situation where it is desirable * This class is useful in a "simple usage" situation where it is desirable
* to be able to get and set some of the tag information that is similar * to be able to get and set some of the tag information that is similar across file types.
* across file types.
* *
* Also note that it is probably a good idea to plug this into your mime * Also note that it is probably a good idea to plug this into your mime
* type system rather than using the constructor that accepts a file name using * type system rather than using the constructor that accepts a file name using the FileTypeResolver.
* the FileTypeResolver.
* *
* \see FileTypeResolver * \see FileTypeResolver
* \see addFileTypeResolver() * \see addFileTypeResolver()
*/ */
class TAGLIB_EXPORT FileRef class TAGLIB_EXPORT FileRef {
{ public:
public:
//! A class for pluggable file type resolution. //! A class for pluggable file type resolution.
/*! /*!
* This class is used to add extend TagLib's very basic file name based file * This class is used to add extend TagLib's very basic file name based file type resolution.
* type resolution.
* *
* This can be accomplished with: * This can be accomplished with:
* *
@@ -85,205 +80,204 @@ namespace TagLib {
* *
* \endcode * \endcode
* *
* Naturally a less contrived example would be slightly more complex. This * Naturally a less contrived example would be slightly more complex.
* can be used to plug in mime-type detection systems or to add new file types * This can be used to plug in mime-type detection systems or to add new file types to TagLib.
* to TagLib.
*/ */
class TAGLIB_EXPORT FileTypeResolver class TAGLIB_EXPORT FileTypeResolver {
{ public:
TAGLIB_IGNORE_MISSING_DESTRUCTOR virtual ~FileTypeResolver() {}
public:
/*!
* This method must be overridden to provide an additional file type
* resolver. If the resolver is able to determine the file type it should
* return a valid File object; if not it should return 0.
*
* \note The created file is then owned by the FileRef and should not be
* deleted. Deletion will happen automatically when the FileRef passes
* out of scope.
*/
virtual File *createFile(FileName fileName,
bool readAudioProperties = true,
AudioProperties::ReadStyle
audioPropertiesStyle = AudioProperties::Average) const = 0;
};
/*! /*!
* Creates a null FileRef. * This method must be overridden to provide an additional file type resolver.
*/ * If the resolver is able to determine the file type it should return a valid File object; if not it should return 0.
FileRef();
/*!
* Create a FileRef from \a fileName. If \a readAudioProperties is true then
* the audio properties will be read using \a audioPropertiesStyle. If
* \a readAudioProperties is false then \a audioPropertiesStyle will be
* ignored.
* *
* Also see the note in the class documentation about why you may not want to * \note The created file is then owned by the FileRef and should not be deleted.
* use this method in your application. * Deletion will happen automatically when the FileRef passes out of scope.
*/ */
explicit FileRef(FileName fileName, virtual File *createFile(FileName fileName, bool readAudioProperties = true, AudioProperties::ReadStyle audioPropertiesStyle = AudioProperties::Average) const = 0;
bool readAudioProperties = true,
AudioProperties::ReadStyle
audioPropertiesStyle = AudioProperties::Average);
/*!
* Construct a FileRef from an opened \a IOStream. If \a readAudioProperties
* is true then the audio properties will be read using \a audioPropertiesStyle.
* If \a readAudioProperties is false then \a audioPropertiesStyle will be
* ignored.
*
* Also see the note in the class documentation about why you may not want to
* use this method in your application.
*
* \note TagLib will *not* take ownership of the stream, the caller is
* responsible for deleting it after the File object.
*/
explicit FileRef(IOStream* stream,
bool readAudioProperties = true,
AudioProperties::ReadStyle
audioPropertiesStyle = AudioProperties::Average);
/*!
* Construct a FileRef using \a file. The FileRef now takes ownership of the
* pointer and will delete the File when it passes out of scope.
*/
explicit FileRef(File *file);
/*!
* Make a copy of \a ref.
*/
FileRef(const FileRef &ref);
/*!
* Destroys this FileRef instance.
*/
virtual ~FileRef();
/*!
* Returns a pointer to represented file's tag.
*
* \warning This pointer will become invalid when this FileRef and all
* copies pass out of scope.
*
* \warning Do not cast it to any subclasses of \class Tag.
* Use tag returning methods of appropriate subclasses of \class File instead.
*
* \see File::tag()
*/
Tag *tag() const;
/*!
* Returns the audio properties for this FileRef. If no audio properties
* were read then this will returns a null pointer.
*/
AudioProperties *audioProperties() const;
/*!
* Returns a pointer to the file represented by this handler class.
*
* As a general rule this call should be avoided since if you need to work
* with file objects directly, you are probably better served instantiating
* the File subclasses (i.e. MPEG::File) manually and working with their APIs.
*
* This <i>handle</i> exists to provide a minimal, generic and value-based
* wrapper around a File. Accessing the file directly generally indicates
* a moving away from this simplicity (and into things beyond the scope of
* FileRef).
*
* \warning This pointer will become invalid when this FileRef and all
* copies pass out of scope.
*/
File *file() const;
/*!
* Saves the file. Returns true on success.
*/
bool save();
/*!
* Adds a FileTypeResolver to the list of those used by TagLib. Each
* additional FileTypeResolver is added to the front of a list of resolvers
* that are tried. If the FileTypeResolver returns zero the next resolver
* is tried.
*
* Returns a pointer to the added resolver (the same one that's passed in --
* this is mostly so that static initializers have something to use for
* assignment).
*
* \see FileTypeResolver
*/
static const FileTypeResolver *addFileTypeResolver(const FileTypeResolver *resolver);
/*!
* As is mentioned elsewhere in this class's documentation, the default file
* type resolution code provided by TagLib only works by comparing file
* extensions.
*
* This method returns the list of file extensions that are used by default.
*
* The extensions are all returned in lowercase, though the comparison used
* by TagLib for resolution is case-insensitive.
*
* \note This does not account for any additional file type resolvers that
* are plugged in. Also note that this is not intended to replace a proper
* mime-type resolution system, but is just here for reference.
*
* \see FileTypeResolver
*/
static StringList defaultFileExtensions();
/*!
* Returns true if the file (and as such other pointers) are null.
*/
bool isNull() const;
/*!
* Assign the file pointed to by \a ref to this FileRef.
*/
FileRef &operator=(const FileRef &ref);
/*!
* Exchanges the content of the FileRef by the content of \a ref.
*/
void swap(FileRef &ref);
/*!
* Returns true if this FileRef and \a ref point to the same File object.
*/
bool operator==(const FileRef &ref) const;
/*!
* Returns true if this FileRef and \a ref do not point to the same File
* object.
*/
bool operator!=(const FileRef &ref) const;
/*!
* A simple implementation of file type guessing. If \a readAudioProperties
* is true then the audio properties will be read using
* \a audioPropertiesStyle. If \a readAudioProperties is false then
* \a audioPropertiesStyle will be ignored.
*
* \note You generally shouldn't use this method, but instead the constructor
* directly.
*
* \deprecated
*/
static File *create(FileName fileName,
bool readAudioProperties = true,
AudioProperties::ReadStyle audioPropertiesStyle = AudioProperties::Average);
private:
void parse(FileName fileName, bool readAudioProperties, AudioProperties::ReadStyle audioPropertiesStyle);
void parse(IOStream *stream, bool readAudioProperties, AudioProperties::ReadStyle audioPropertiesStyle);
class FileRefPrivate;
FileRefPrivate *d;
}; };
} /*!
} // namespace Strawberry_TagLib::TagLib * Creates a null FileRef.
*/
explicit FileRef();
#endif /*!
* Create a FileRef from \a fileName.
* If \a readAudioProperties is true then the audio properties will be read using \a audioPropertiesStyle.
* If \a readAudioProperties is false then \a audioPropertiesStyle will be ignored.
*
* Also see the note in the class documentation about why you may not want to
* use this method in your application.
*/
explicit FileRef(FileName fileName, bool readAudioProperties = true, AudioProperties::ReadStyle audioPropertiesStyle = AudioProperties::Average);
/*!
* Construct a FileRef from an opened \a IOStream.
* If \a readAudioProperties is true then the audio properties will be read using \a audioPropertiesStyle.
* If \a readAudioProperties is false then \a audioPropertiesStyle will be ignored.
*
* Also see the note in the class documentation about why you may not want to use this method in your application.
*
* \note TagLib will *not* take ownership of the stream, the caller is responsible for deleting it after the File object.
*/
explicit FileRef(IOStream *stream, bool readAudioProperties = true, AudioProperties::ReadStyle audioPropertiesStyle = AudioProperties::Average);
/*!
* Construct a FileRef using \a file.
* The FileRef now takes ownership of the pointer and will delete the File when it passes out of scope.
*/
explicit FileRef(File *file);
/*!
* Make a copy of \a ref.
*/
FileRef(const FileRef &ref);
/*!
* Destroys this FileRef instance.
*/
virtual ~FileRef();
/*!
* Returns a pointer to represented file's tag.
*
* \warning This pointer will become invalid when this FileRef and all
* copies pass out of scope.
*
* \warning Do not cast it to any subclasses of \class Tag.
* Use tag returning methods of appropriate subclasses of \class File instead.
*
* \see File::tag()
*/
Tag *tag() const;
/*!
* Exports the tags of the file as dictionary mapping (human readable) tag names (uppercase Strings) to StringLists of tag values.
* Calls the according specialization in the File subclasses.
* For each metadata object of the file that could not be parsed into the PropertyMap format,
* the returend map's unsupportedData() list will contain one entry identifying that object (e.g. the frame type for ID3v2 tags).
* Use removeUnsupportedProperties() to remove (a subset of) them.
* For files that contain more than one tag (e.g. an MP3 with both an ID3v1 and an ID3v2 tag) only the most "modern" one will be exported (ID3v2 in this case).
*/
PropertyMap properties() const;
/*!
* Removes unsupported properties, or a subset of them, from the file's metadata.
* The parameter \a properties must contain only entries from properties().unsupportedData().
*/
void removeUnsupportedProperties(const StringList &properties);
/*!
* Sets the tags of this File to those specified in \a properties.
* Calls the according specialization method in the subclasses of File to do the translation into the format-specific details.
* If some value(s) could not be written imported to the specific metadata format,
* the returned PropertyMap will contain those value(s). Otherwise it will be empty, indicating that no problems occured.
* With file types that support several tag formats (for instance, MP3 files can have ID3v1, ID3v2, and APEv2 tags),
* this function will create the most appropriate one (ID3v2 for MP3 files). Older formats will be updated as well,
* if they exist, but won't be taken into account for the return value of this function.
* See the documentation of the subclass implementations for detailed descriptions.
*/
PropertyMap setProperties(const PropertyMap &properties);
/*!
* Returns the audio properties for this FileRef.
* If no audio properties were read then this will returns a null pointer.
*/
AudioProperties *audioProperties() const;
/*!
* Returns a pointer to the file represented by this handler class.
*
* As a general rule this call should be avoided since if you need to work
* with file objects directly, you are probably better served instantiating
* the File subclasses (i.e. MPEG::File) manually and working with their APIs.
*
* This <i>handle</i> exists to provide a minimal, generic and value-based
* wrapper around a File. Accessing the file directly generally indicates
* a moving away from this simplicity (and into things beyond the scope of
* FileRef).
*
* \warning This pointer will become invalid when this FileRef and all
* copies pass out of scope.
*/
File *file() const;
/*!
* Saves the file. Returns true on success.
*/
bool save();
/*!
* Adds a FileTypeResolver to the list of those used by TagLib.
* Each additional FileTypeResolver is added to the front of a list of resolvers that are tried.
* If the FileTypeResolver returns zero the next resolver is tried.
*
* Returns a pointer to the added resolver (the same one that's passed in --
* this is mostly so that static initializers have something to use for assignment).
*
* \see FileTypeResolver
*/
static const FileTypeResolver *addFileTypeResolver(const FileTypeResolver *resolver);
/*!
* As is mentioned elsewhere in this class's documentation, the default file
* type resolution code provided by TagLib only works by comparing file extensions.
*
* This method returns the list of file extensions that are used by default.
*
* The extensions are all returned in lowercase, though the comparison used
* by TagLib for resolution is case-insensitive.
*
* \note This does not account for any additional file type resolvers that
* are plugged in. Also note that this is not intended to replace a proper
* mime-type resolution system, but is just here for reference.
*
* \see FileTypeResolver
*/
static StringList defaultFileExtensions();
/*!
* Returns true if the file is open and readable.
*
* \note Just a negative of isNull().
*/
bool isValid() const;
/*!
* Returns true if the file (and as such other pointers) are null.
*/
bool isNull() const;
/*!
* Assign the file pointed to by \a ref to this FileRef.
*/
FileRef &operator=(const FileRef &ref);
/*!
* Exchanges the content of the FileRef by the content of \a ref.
*/
void swap(FileRef &ref);
/*!
* Returns true if this FileRef and \a ref point to the same File object.
*/
bool operator==(const FileRef &ref) const;
/*!
* Returns true if this FileRef and \a ref do not point to the same File object.
*/
bool operator!=(const FileRef &ref) const;
private:
void parse(FileName fileName, bool readAudioProperties, AudioProperties::ReadStyle audioPropertiesStyle);
void parse(IOStream *stream, bool readAudioProperties, AudioProperties::ReadStyle audioPropertiesStyle);
class FileRefPrivate;
FileRefPrivate *d;
};
} // namespace TagLib
} // namespace Strawberry_TagLib
#endif // TAGLIB_FILEREF_H

View File

@@ -23,18 +23,20 @@
* http://www.mozilla.org/MPL/ * * http://www.mozilla.org/MPL/ *
***************************************************************************/ ***************************************************************************/
#include <tbytevector.h> #include <memory>
#include <tstring.h>
#include <tlist.h>
#include <tdebug.h>
#include <tagunion.h>
#include <tpropertymap.h>
#include <tagutils.h>
#include <id3v2header.h> #include "tbytevector.h"
#include <id3v2tag.h> #include "tstring.h"
#include <id3v1tag.h> #include "tlist.h"
#include <xiphcomment.h> #include "tdebug.h"
#include "tagunion.h"
#include "tpropertymap.h"
#include "tagutils.h"
#include "id3v2header.h"
#include "id3v2tag.h"
#include "id3v1tag.h"
#include "xiphcomment.h"
#include "flacpicture.h" #include "flacpicture.h"
#include "flacfile.h" #include "flacfile.h"
@@ -43,55 +45,63 @@
using namespace Strawberry_TagLib::TagLib; using namespace Strawberry_TagLib::TagLib;
namespace namespace {
{ typedef List<std::shared_ptr<FLAC::MetadataBlock>> BlockList;
typedef List<FLAC::MetadataBlock *> BlockList; typedef BlockList::Iterator BlockIterator;
typedef BlockList::Iterator BlockIterator; typedef BlockList::Iterator BlockConstIterator;
typedef BlockList::Iterator BlockConstIterator;
enum { FlacXiphIndex = 0, FlacID3v2Index = 1, FlacID3v1Index = 2 }; enum { FlacXiphIndex = 0,
FlacID3v2Index = 1,
FlacID3v1Index = 2 };
const long MinPaddingLength = 4096; const long long MinPaddingLength = 4096;
const long MaxPaddingLegnth = 1024 * 1024; const long long MaxPaddingLegnth = 1024 * 1024;
const char LastBlockFlag = '\x80'; const char LastBlockFlag = '\x80';
} // namespace
namespace Strawberry_TagLib {
namespace TagLib {
namespace FLAC {
// Enables BlockList::find() to take raw pointers.
bool operator==(std::shared_ptr<MetadataBlock> lhs, MetadataBlock *rhs);
bool operator==(std::shared_ptr<MetadataBlock> lhs, MetadataBlock *rhs) {
return lhs.get() == rhs;
} }
} // namespace FLAC
} // namespace TagLib
} // namespace Strawberry_TagLib
class FLAC::File::FilePrivate class FLAC::File::FilePrivate {
{ public:
public: explicit FilePrivate(const ID3v2::FrameFactory *frameFactory) : ID3v2FrameFactory(ID3v2::FrameFactory::instance()),
explicit FilePrivate(const ID3v2::FrameFactory *frameFactory = ID3v2::FrameFactory::instance()) : ID3v2Location(-1),
ID3v2FrameFactory(frameFactory), ID3v2OriginalSize(0),
ID3v2Location(-1), ID3v1Location(-1),
ID3v2OriginalSize(0), flacStart(0),
ID3v1Location(-1), streamStart(0),
properties(0), scanned(false) {
flacStart(0),
streamStart(0), if (frameFactory)
scanned(false) ID3v2FrameFactory = frameFactory;
{
blocks.setAutoDelete(true);
}
~FilePrivate()
{
delete properties;
} }
const ID3v2::FrameFactory *ID3v2FrameFactory; const ID3v2::FrameFactory *ID3v2FrameFactory;
long ID3v2Location; long long ID3v2Location;
long ID3v2OriginalSize; long long ID3v2OriginalSize;
long ID3v1Location; long long ID3v1Location;
TagUnion tag; TripleTagUnion tag;
Properties *properties; std::unique_ptr<AudioProperties> properties;
ByteVector xiphCommentData; ByteVector xiphCommentData;
BlockList blocks; BlockList blocks;
long flacStart; long long flacStart;
long streamStart; long long streamStart;
bool scanned; bool scanned;
}; };
@@ -99,111 +109,85 @@ public:
// static members // static members
//////////////////////////////////////////////////////////////////////////////// ////////////////////////////////////////////////////////////////////////////////
bool FLAC::File::isSupported(IOStream *stream) bool FLAC::File::isSupported(IOStream *stream) {
{
// A FLAC file has an ID "fLaC" somewhere. An ID3v2 tag may precede. // A FLAC file has an ID "fLaC" somewhere. An ID3v2 tag may precede.
const ByteVector buffer = Utils::readHeader(stream, bufferSize(), true); const ByteVector buffer = Utils::readHeader(stream, bufferSize(), true);
return (buffer.find("fLaC") >= 0); return (buffer.find("fLaC") != ByteVector::npos());
} }
//////////////////////////////////////////////////////////////////////////////// ////////////////////////////////////////////////////////////////////////////////
// public members // public members
//////////////////////////////////////////////////////////////////////////////// ////////////////////////////////////////////////////////////////////////////////
FLAC::File::File(FileName file, bool readProperties, Properties::ReadStyle) : FLAC::File::File(FileName file, bool readProperties, AudioProperties::ReadStyle, ID3v2::FrameFactory *frameFactory) : Strawberry_TagLib::TagLib::File(file), d(new FilePrivate(frameFactory)) {
Strawberry_TagLib::TagLib::File(file),
d(new FilePrivate()) if (isOpen())
{
if(isOpen())
read(readProperties); read(readProperties);
} }
FLAC::File::File(FileName file, ID3v2::FrameFactory *frameFactory, FLAC::File::File(IOStream *stream, bool readProperties, AudioProperties::ReadStyle, ID3v2::FrameFactory *frameFactory) : Strawberry_TagLib::TagLib::File(stream), d(new FilePrivate(frameFactory)) {
bool readProperties, Properties::ReadStyle) :
Strawberry_TagLib::TagLib::File(file), if (isOpen())
d(new FilePrivate(frameFactory))
{
if(isOpen())
read(readProperties); read(readProperties);
} }
FLAC::File::File(IOStream *stream, ID3v2::FrameFactory *frameFactory, FLAC::File::~File() {
bool readProperties, Properties::ReadStyle) :
Strawberry_TagLib::TagLib::File(stream),
d(new FilePrivate(frameFactory))
{
if(isOpen())
read(readProperties);
}
FLAC::File::~File()
{
delete d; delete d;
} }
Strawberry_TagLib::TagLib::Tag *FLAC::File::tag() const Strawberry_TagLib::TagLib::Tag *FLAC::File::tag() const {
{
return &d->tag; return &d->tag;
} }
PropertyMap FLAC::File::properties() const PropertyMap FLAC::File::setProperties(const PropertyMap &properties) {
{
return d->tag.properties();
}
void FLAC::File::removeUnsupportedProperties(const StringList &unsupported)
{
d->tag.removeUnsupportedProperties(unsupported);
}
PropertyMap FLAC::File::setProperties(const PropertyMap &properties)
{
return xiphComment(true)->setProperties(properties); return xiphComment(true)->setProperties(properties);
} }
FLAC::Properties *FLAC::File::audioProperties() const FLAC::AudioProperties *FLAC::File::audioProperties() const {
{ return d->properties.get();
return d->properties;
} }
bool FLAC::File::save() bool FLAC::File::save() {
{
if(readOnly()) { if (readOnly()) {
debug("FLAC::File::save() - Cannot save to a read only file."); debug("FLAC::File::save() - Cannot save to a read only file.");
return false; return false;
} }
if(!isValid()) { if (!isValid()) {
debug("FLAC::File::save() -- Trying to save invalid file."); debug("FLAC::File::save() -- Trying to save invalid file.");
return false; return false;
} }
// Create new vorbis comments // Create new vorbis comments
if(!hasXiphComment()) if (!hasXiphComment())
Tag::duplicate(&d->tag, xiphComment(true), false); Tag::duplicate(&d->tag, xiphComment(true), false);
d->xiphCommentData = xiphComment()->render(false); d->xiphCommentData = xiphComment()->render(false);
// Replace metadata blocks // Replace metadata blocks
for(BlockIterator it = d->blocks.begin(); it != d->blocks.end(); ++it) { for (BlockIterator it = d->blocks.begin(); it != d->blocks.end(); ++it) {
if((*it)->code() == MetadataBlock::VorbisComment) { if ((*it)->code() == MetadataBlock::VorbisComment) {
// Set the new Vorbis Comment block // Set the new Vorbis Comment block
delete *it;
d->blocks.erase(it); d->blocks.erase(it);
break; break;
} }
} }
d->blocks.append(new UnknownMetadataBlock(MetadataBlock::VorbisComment, d->xiphCommentData)); d->blocks.append(std::shared_ptr<MetadataBlock>(new UnknownMetadataBlock(MetadataBlock::VorbisComment, d->xiphCommentData)));
// Render data for the metadata blocks // Render data for the metadata blocks
ByteVector data; ByteVector data;
for(BlockConstIterator it = d->blocks.begin(); it != d->blocks.end(); ++it) { for (BlockConstIterator it = d->blocks.begin(); it != d->blocks.end(); ++it) {
ByteVector blockData = (*it)->render(); ByteVector blockData = (*it)->render();
ByteVector blockHeader = ByteVector::fromUInt(blockData.size()); ByteVector blockHeader = ByteVector::fromUInt32BE(blockData.size());
blockHeader[0] = (*it)->code(); blockHeader[0] = (*it)->code();
data.append(blockHeader); data.append(blockHeader);
data.append(blockData); data.append(blockData);
@@ -211,27 +195,27 @@ bool FLAC::File::save()
// Compute the amount of padding, and append that to data. // Compute the amount of padding, and append that to data.
long originalLength = d->streamStart - d->flacStart; long long originalLength = d->streamStart - d->flacStart;
long paddingLength = originalLength - data.size() - 4; long long paddingLength = originalLength - data.size() - 4;
if(paddingLength <= 0) { if (paddingLength <= 0) {
paddingLength = MinPaddingLength; paddingLength = MinPaddingLength;
} }
else { else {
// Padding won't increase beyond 1% of the file size or 1MB. // Padding won't increase beyond 1% of the file size or 1MB.
long threshold = length() / 100; long long threshold = length() / 100;
threshold = std::max(threshold, MinPaddingLength); threshold = std::max(threshold, MinPaddingLength);
threshold = std::min(threshold, MaxPaddingLegnth); threshold = std::min(threshold, MaxPaddingLegnth);
if(paddingLength > threshold) if (paddingLength > threshold)
paddingLength = MinPaddingLength; paddingLength = MinPaddingLength;
} }
ByteVector paddingHeader = ByteVector::fromUInt(paddingLength); ByteVector paddingHeader = ByteVector::fromUInt32BE(paddingLength);
paddingHeader[0] = static_cast<char>(MetadataBlock::Padding | LastBlockFlag); paddingHeader[0] = static_cast<char>(MetadataBlock::Padding | LastBlockFlag);
data.append(paddingHeader); data.append(paddingHeader);
data.resize(static_cast<unsigned int>(data.size() + paddingLength)); data.resize(static_cast<size_t>(data.size() + paddingLength));
// Write the data to the file // Write the data to the file
@@ -239,25 +223,25 @@ bool FLAC::File::save()
d->streamStart += (static_cast<long>(data.size()) - originalLength); d->streamStart += (static_cast<long>(data.size()) - originalLength);
if(d->ID3v1Location >= 0) if (d->ID3v1Location >= 0)
d->ID3v1Location += (static_cast<long>(data.size()) - originalLength); d->ID3v1Location += (static_cast<long>(data.size()) - originalLength);
// Update ID3 tags // Update ID3 tags
if(ID3v2Tag() && !ID3v2Tag()->isEmpty()) { if (ID3v2Tag() && !ID3v2Tag()->isEmpty()) {
// ID3v2 tag is not empty. Update the old one or create a new one. // ID3v2 tag is not empty. Update the old one or create a new one.
if(d->ID3v2Location < 0) if (d->ID3v2Location < 0)
d->ID3v2Location = 0; d->ID3v2Location = 0;
data = ID3v2Tag()->render(); data = ID3v2Tag()->render();
insert(data, d->ID3v2Location, d->ID3v2OriginalSize); insert(data, d->ID3v2Location, d->ID3v2OriginalSize);
d->flacStart += (static_cast<long>(data.size()) - d->ID3v2OriginalSize); d->flacStart += (static_cast<long>(data.size()) - d->ID3v2OriginalSize);
d->streamStart += (static_cast<long>(data.size()) - d->ID3v2OriginalSize); d->streamStart += (static_cast<long>(data.size()) - d->ID3v2OriginalSize);
if(d->ID3v1Location >= 0) if (d->ID3v1Location >= 0)
d->ID3v1Location += (static_cast<long>(data.size()) - d->ID3v2OriginalSize); d->ID3v1Location += (static_cast<long>(data.size()) - d->ID3v2OriginalSize);
d->ID3v2OriginalSize = data.size(); d->ID3v2OriginalSize = data.size();
@@ -266,13 +250,13 @@ bool FLAC::File::save()
// ID3v2 tag is empty. Remove the old one. // ID3v2 tag is empty. Remove the old one.
if(d->ID3v2Location >= 0) { if (d->ID3v2Location >= 0) {
removeBlock(d->ID3v2Location, d->ID3v2OriginalSize); removeBlock(d->ID3v2Location, d->ID3v2OriginalSize);
d->flacStart -= d->ID3v2OriginalSize; d->flacStart -= d->ID3v2OriginalSize;
d->streamStart -= d->ID3v2OriginalSize; d->streamStart -= d->ID3v2OriginalSize;
if(d->ID3v1Location >= 0) if (d->ID3v1Location >= 0)
d->ID3v1Location -= d->ID3v2OriginalSize; d->ID3v1Location -= d->ID3v2OriginalSize;
d->ID3v2Location = -1; d->ID3v2Location = -1;
@@ -280,11 +264,11 @@ bool FLAC::File::save()
} }
} }
if(ID3v1Tag() && !ID3v1Tag()->isEmpty()) { if (ID3v1Tag() && !ID3v1Tag()->isEmpty()) {
// ID3v1 tag is not empty. Update the old one or create a new one. // ID3v1 tag is not empty. Update the old one or create a new one.
if(d->ID3v1Location >= 0) { if (d->ID3v1Location >= 0) {
seek(d->ID3v1Location); seek(d->ID3v1Location);
} }
else { else {
@@ -298,113 +282,90 @@ bool FLAC::File::save()
// ID3v1 tag is empty. Remove the old one. // ID3v1 tag is empty. Remove the old one.
if(d->ID3v1Location >= 0) { if (d->ID3v1Location >= 0) {
truncate(d->ID3v1Location); truncate(d->ID3v1Location);
d->ID3v1Location = -1; d->ID3v1Location = -1;
} }
} }
return true; return true;
} }
ID3v2::Tag *FLAC::File::ID3v2Tag(bool create) ID3v2::Tag *FLAC::File::ID3v2Tag(bool create) {
{
return d->tag.access<ID3v2::Tag>(FlacID3v2Index, create); return d->tag.access<ID3v2::Tag>(FlacID3v2Index, create);
} }
ID3v1::Tag *FLAC::File::ID3v1Tag(bool create) ID3v1::Tag *FLAC::File::ID3v1Tag(bool create) {
{
return d->tag.access<ID3v1::Tag>(FlacID3v1Index, create); return d->tag.access<ID3v1::Tag>(FlacID3v1Index, create);
} }
Ogg::XiphComment *FLAC::File::xiphComment(bool create) Ogg::XiphComment *FLAC::File::xiphComment(bool create) {
{
return d->tag.access<Ogg::XiphComment>(FlacXiphIndex, create); return d->tag.access<Ogg::XiphComment>(FlacXiphIndex, create);
} }
void FLAC::File::setID3v2FrameFactory(const ID3v2::FrameFactory *factory) List<FLAC::Picture *> FLAC::File::pictureList() {
{
d->ID3v2FrameFactory = factory;
}
ByteVector FLAC::File::streamInfoData()
{
debug("FLAC::File::streamInfoData() -- This function is obsolete. Returning an empty ByteVector.");
return ByteVector();
}
long FLAC::File::streamLength()
{
debug("FLAC::File::streamLength() -- This function is obsolete. Returning zero.");
return 0;
}
List<FLAC::Picture *> FLAC::File::pictureList()
{
List<Picture *> pictures; List<Picture *> pictures;
for(BlockConstIterator it = d->blocks.begin(); it != d->blocks.end(); ++it) { for (BlockConstIterator it = d->blocks.begin(); it != d->blocks.end(); ++it) {
Picture *picture = dynamic_cast<Picture *>(*it); Picture *picture = dynamic_cast<Picture *>(it->get());
if(picture) { if (picture) {
pictures.append(picture); pictures.append(picture);
} }
} }
return pictures; return pictures;
} }
void FLAC::File::addPicture(Picture *picture) void FLAC::File::addPicture(Picture *picture) {
{ d->blocks.append(std::shared_ptr<Picture>(picture));
d->blocks.append(picture);
} }
void FLAC::File::removePicture(Picture *picture, bool del) void FLAC::File::removePicture(Picture *picture) {
{
BlockIterator it = d->blocks.find(picture); BlockIterator it = d->blocks.find(picture);
if(it != d->blocks.end()) if (it != d->blocks.end())
d->blocks.erase(it); d->blocks.erase(it);
if(del)
delete picture;
} }
void FLAC::File::removePictures() void FLAC::File::removePictures() {
{
for(BlockIterator it = d->blocks.begin(); it != d->blocks.end(); ) { for (BlockIterator it = d->blocks.begin(); it != d->blocks.end();) {
if(dynamic_cast<Picture *>(*it)) { if (dynamic_cast<Picture *>(it->get())) {
delete *it;
it = d->blocks.erase(it); it = d->blocks.erase(it);
} }
else { else {
++it; ++it;
} }
} }
} }
void FLAC::File::strip(int tags) void FLAC::File::strip(int tags) {
{
if(tags & ID3v1)
d->tag.set(FlacID3v1Index, 0);
if(tags & ID3v2) if (tags & ID3v1)
d->tag.set(FlacID3v2Index, 0); d->tag.set(FlacID3v1Index, nullptr);
if(tags & XiphComment) { if (tags & ID3v2)
d->tag.set(FlacID3v2Index, nullptr);
if (tags & XiphComment) {
xiphComment()->removeAllFields(); xiphComment()->removeAllFields();
xiphComment()->removeAllPictures(); xiphComment()->removeAllPictures();
} }
} }
bool FLAC::File::hasXiphComment() const bool FLAC::File::hasXiphComment() const {
{
return !d->xiphCommentData.isEmpty(); return !d->xiphCommentData.isEmpty();
} }
bool FLAC::File::hasID3v1Tag() const bool FLAC::File::hasID3v1Tag() const {
{
return (d->ID3v1Location >= 0); return (d->ID3v1Location >= 0);
} }
bool FLAC::File::hasID3v2Tag() const bool FLAC::File::hasID3v2Tag() const {
{
return (d->ID3v2Location >= 0); return (d->ID3v2Location >= 0);
} }
@@ -412,13 +373,13 @@ bool FLAC::File::hasID3v2Tag() const
// private members // private members
//////////////////////////////////////////////////////////////////////////////// ////////////////////////////////////////////////////////////////////////////////
void FLAC::File::read(bool readProperties) void FLAC::File::read(bool readProperties) {
{
// Look for an ID3v2 tag // Look for an ID3v2 tag
d->ID3v2Location = Utils::findID3v2(this); d->ID3v2Location = Utils::findID3v2(this);
if(d->ID3v2Location >= 0) { if (d->ID3v2Location >= 0) {
d->tag.set(FlacID3v2Index, new ID3v2::Tag(this, d->ID3v2Location, d->ID3v2FrameFactory)); d->tag.set(FlacID3v2Index, new ID3v2::Tag(this, d->ID3v2Location, d->ID3v2FrameFactory));
d->ID3v2OriginalSize = ID3v2Tag()->header()->completeTagSize(); d->ID3v2OriginalSize = ID3v2Tag()->header()->completeTagSize();
} }
@@ -427,56 +388,57 @@ void FLAC::File::read(bool readProperties)
d->ID3v1Location = Utils::findID3v1(this); d->ID3v1Location = Utils::findID3v1(this);
if(d->ID3v1Location >= 0) if (d->ID3v1Location >= 0)
d->tag.set(FlacID3v1Index, new ID3v1::Tag(this, d->ID3v1Location)); d->tag.set(FlacID3v1Index, new ID3v1::Tag(this, d->ID3v1Location));
// Look for FLAC metadata, including vorbis comments // Look for FLAC metadata, including vorbis comments
scan(); scan();
if(!isValid()) if (!isValid())
return; return;
if(!d->xiphCommentData.isEmpty()) if (!d->xiphCommentData.isEmpty())
d->tag.set(FlacXiphIndex, new Ogg::XiphComment(d->xiphCommentData)); d->tag.set(FlacXiphIndex, new Ogg::XiphComment(d->xiphCommentData));
else else
d->tag.set(FlacXiphIndex, new Ogg::XiphComment()); d->tag.set(FlacXiphIndex, new Ogg::XiphComment());
if(readProperties) { if (readProperties) {
// First block should be the stream_info metadata // First block should be the stream_info metadata
const ByteVector infoData = d->blocks.front()->render(); const ByteVector infoData = d->blocks.front()->render();
long streamLength; long long streamLength;
if(d->ID3v1Location >= 0) if (d->ID3v1Location >= 0)
streamLength = d->ID3v1Location - d->streamStart; streamLength = d->ID3v1Location - d->streamStart;
else else
streamLength = length() - d->streamStart; streamLength = length() - d->streamStart;
d->properties = new Properties(infoData, streamLength); d->properties.reset(new AudioProperties(infoData, streamLength));
} }
} }
void FLAC::File::scan() void FLAC::File::scan() {
{
// Scan the metadata pages // Scan the metadata pages
if(d->scanned) if (d->scanned)
return; return;
if(!isValid()) if (!isValid())
return; return;
long nextBlockOffset; long long nextBlockOffset;
if(d->ID3v2Location >= 0) if (d->ID3v2Location >= 0)
nextBlockOffset = find("fLaC", d->ID3v2Location + d->ID3v2OriginalSize); nextBlockOffset = find("fLaC", d->ID3v2Location + d->ID3v2OriginalSize);
else else
nextBlockOffset = find("fLaC"); nextBlockOffset = find("fLaC");
if(nextBlockOffset < 0) { if (nextBlockOffset < 0) {
debug("FLAC::File::scan() -- FLAC stream not found"); debug("FLAC::File::scan() -- FLAC stream not found");
setValid(false); setValid(false);
return; return;
@@ -485,7 +447,7 @@ void FLAC::File::scan()
nextBlockOffset += 4; nextBlockOffset += 4;
d->flacStart = nextBlockOffset; d->flacStart = nextBlockOffset;
while(true) { while (true) {
seek(nextBlockOffset); seek(nextBlockOffset);
const ByteVector header = readBlock(4); const ByteVector header = readBlock(4);
@@ -504,66 +466,63 @@ void FLAC::File::scan()
const char blockType = header[0] & ~LastBlockFlag; const char blockType = header[0] & ~LastBlockFlag;
const bool isLastBlock = (header[0] & LastBlockFlag) != 0; const bool isLastBlock = (header[0] & LastBlockFlag) != 0;
const unsigned int blockLength = header.toUInt(1U, 3U); const size_t blockLength = header.toUInt24BE(1);
// First block should be the stream_info metadata // First block should be the stream_info metadata
if(d->blocks.isEmpty() && blockType != MetadataBlock::StreamInfo) { if (d->blocks.isEmpty() && blockType != MetadataBlock::StreamInfo) {
debug("FLAC::File::scan() -- First block should be the stream_info metadata"); debug("FLAC::File::scan() -- First block should be the stream_info metadata");
setValid(false); setValid(false);
return; return;
} }
if(blockLength == 0 if (blockLength == 0 && blockType != MetadataBlock::Padding && blockType != MetadataBlock::SeekTable) {
&& blockType != MetadataBlock::Padding && blockType != MetadataBlock::SeekTable)
{
debug("FLAC::File::scan() -- Zero-sized metadata block found"); debug("FLAC::File::scan() -- Zero-sized metadata block found");
setValid(false); setValid(false);
return; return;
} }
const ByteVector data = readBlock(blockLength); const ByteVector data = readBlock(blockLength);
if(data.size() != blockLength) { if (data.size() != blockLength) {
debug("FLAC::File::scan() -- Failed to read a metadata block"); debug("FLAC::File::scan() -- Failed to read a metadata block");
setValid(false); setValid(false);
return; return;
} }
MetadataBlock *block = 0; std::shared_ptr<MetadataBlock> block;
// Found the vorbis-comment // Found the vorbis-comment
if(blockType == MetadataBlock::VorbisComment) { if (blockType == MetadataBlock::VorbisComment) {
if(d->xiphCommentData.isEmpty()) { if (d->xiphCommentData.isEmpty()) {
d->xiphCommentData = data; d->xiphCommentData = data;
block = new UnknownMetadataBlock(MetadataBlock::VorbisComment, data); block.reset(new UnknownMetadataBlock(MetadataBlock::VorbisComment, data));
} }
else { else {
debug("FLAC::File::scan() -- multiple Vorbis Comment blocks found, discarding"); debug("FLAC::File::scan() -- multiple Vorbis Comment blocks found, discarding");
} }
} }
else if(blockType == MetadataBlock::Picture) { else if (blockType == MetadataBlock::Picture) {
FLAC::Picture *picture = new FLAC::Picture(); std::shared_ptr<FLAC::Picture> picture(new FLAC::Picture());
if(picture->parse(data)) { if (picture->parse(data)) {
block = picture; block = picture;
} }
else { else {
debug("FLAC::File::scan() -- invalid picture found, discarding"); debug("FLAC::File::scan() -- invalid picture found, discarding");
delete picture;
} }
} }
else if(blockType == MetadataBlock::Padding) { else if (blockType == MetadataBlock::Padding) {
// Skip all padding blocks. // Skip all padding blocks.
} }
else { else {
block = new UnknownMetadataBlock(blockType, data); block.reset(new UnknownMetadataBlock(blockType, data));
} }
if(block) if (block)
d->blocks.append(block); d->blocks.append(block);
nextBlockOffset += blockLength + 4; nextBlockOffset += blockLength + 4;
if(isLastBlock) if (isLastBlock)
break; break;
} }
@@ -572,4 +531,5 @@ void FLAC::File::scan()
d->streamStart = nextBlockOffset; d->streamStart = nextBlockOffset;
d->scanned = true; d->scanned = true;
} }

View File

@@ -26,6 +26,8 @@
#ifndef TAGLIB_FLACFILE_H #ifndef TAGLIB_FLACFILE_H
#define TAGLIB_FLACFILE_H #define TAGLIB_FLACFILE_H
#include <memory>
#include "taglib_export.h" #include "taglib_export.h"
#include "tfile.h" #include "tfile.h"
#include "tlist.h" #include "tlist.h"
@@ -37,309 +39,240 @@
namespace Strawberry_TagLib { namespace Strawberry_TagLib {
namespace TagLib { namespace TagLib {
class Tag; class Tag;
namespace ID3v2 { class FrameFactory; class Tag; } namespace ID3v2 {
namespace ID3v1 { class Tag; } class FrameFactory;
namespace Ogg { class XiphComment; } class Tag;
} // namespace ID3v2
namespace ID3v1 {
class Tag;
}
namespace Ogg {
class XiphComment;
}
//! An implementation of FLAC metadata //! An implementation of FLAC metadata
/*!
* This is implementation of FLAC metadata for non-Ogg FLAC files. At some
* point when Ogg / FLAC is more common there will be a similar implementation
* under the Ogg hierarchy.
*
* This supports ID3v1, ID3v2 and Xiph style comments as well as reading stream
* properties from the file.
*/
namespace FLAC {
//! An implementation of TagLib::File with FLAC specific methods
/*!
* This implements and provides an interface for FLAC files to the TagLib::Tag and TagLib::AudioProperties interfaces
* by way of implementing the abstract TagLib::File API as well as providing some additional information specific to FLAC files.
*
*/
class TAGLIB_EXPORT File : public Strawberry_TagLib::TagLib::File {
public:
/*!
* This set of flags is used for various operations and is suitable for
* being OR-ed together.
*/
enum TagTypes {
//! Empty set. Matches no tag types.
NoTags = 0x0000,
//! Matches Vorbis comments.
XiphComment = 0x0001,
//! Matches ID3v1 tags.
ID3v1 = 0x0002,
//! Matches ID3v2 tags.
ID3v2 = 0x0004,
//! Matches all tag types.
AllTags = 0xffff
};
/*! /*!
* This is implementation of FLAC metadata for non-Ogg FLAC files. At some * Constructs an FLAC file from \a file.
* point when Ogg / FLAC is more common there will be a similar implementation * If \a readProperties is true the file's audio properties will also be read.
* under the Ogg hierarchy.
* *
* This supports ID3v1, ID3v2 and Xiph style comments as well as reading stream * If this file contains and ID3v2 tag the frames will be created using \a frameFactory.
* properties from the file. *
* \note In the current implementation, \a propertiesStyle is ignored.
*/ */
explicit File(FileName file, bool readProperties = true, AudioProperties::ReadStyle propertiesStyle = AudioProperties::Average, ID3v2::FrameFactory *frameFactory = nullptr);
namespace FLAC { /*!
* Constructs a FLAC file from \a stream. If \a readProperties is true the file's audio properties will also be read.
*
* \note TagLib will *not* take ownership of the stream, the caller is responsible for deleting it after the File object.
*
* If this file contains and ID3v2 tag the frames will be created using \a frameFactory.
*
* \note In the current implementation, \a propertiesStyle is ignored.
*/
// BIC: merge with the above constructor
explicit File(IOStream *stream, bool readProperties = true, AudioProperties::ReadStyle propertiesStyle = AudioProperties::Average, ID3v2::FrameFactory *frameFactory = nullptr);
//! An implementation of Strawberry_TagLib::TagLib::File with FLAC specific methods /*!
* Destroys this instance of the File.
*/
~File() override;
/*! /*!
* This implements and provides an interface for FLAC files to the * Returns the Tag for this file. This will be a union of XiphComment, ID3v1 and ID3v2 tags.
* Strawberry_TagLib::TagLib::Tag and Strawberry_TagLib::TagLib::AudioProperties interfaces by way of implementing *
* the abstract Strawberry_TagLib::TagLib::File API as well as providing some additional * \see ID3v2Tag()
* information specific to FLAC files. * \see ID3v1Tag()
*/ * \see XiphComment()
*/
Strawberry_TagLib::TagLib::Tag *tag() const override;
class TAGLIB_EXPORT File : public Strawberry_TagLib::TagLib::File /*!
{ * Implements the unified property interface -- import function.
public: * This always creates a Xiph comment, if none exists. The return value relates to the Xiph comment only.
/*! * Ignores any changes to ID3v1 or ID3v2 comments since they are not allowed in the FLAC specification.
* This set of flags is used for various operations and is suitable for */
* being OR-ed together. PropertyMap setProperties(const PropertyMap &) override;
*/
enum TagTypes {
//! Empty set. Matches no tag types.
NoTags = 0x0000,
//! Matches Vorbis comments.
XiphComment = 0x0001,
//! Matches ID3v1 tags.
ID3v1 = 0x0002,
//! Matches ID3v2 tags.
ID3v2 = 0x0004,
//! Matches all tag types.
AllTags = 0xffff
};
/*! /*!
* Constructs a FLAC file from \a file. If \a readProperties is true the * Returns the FLAC::AudioProperties for this file. If no audio properties were read then this will return a null pointer.
* file's audio properties will also be read. */
* AudioProperties *audioProperties() const override;
* \note In the current implementation, \a propertiesStyle is ignored.
*
* \deprecated This constructor will be dropped in favor of the one below
* in a future version.
*/
File(FileName file, bool readProperties = true,
Properties::ReadStyle propertiesStyle = Properties::Average);
/*! /*!
* Constructs an FLAC file from \a file. If \a readProperties is true the * Save the file. This will primarily save the XiphComment, but will also keep any old ID3-tags up to date.
* file's audio properties will also be read. * If the file has no XiphComment, one will be constructed from the ID3-tags.
* *
* If this file contains and ID3v2 tag the frames will be created using * This returns true if the save was successful.
* \a frameFactory. */
* bool save() override;
* \note In the current implementation, \a propertiesStyle is ignored.
*/
// BIC: merge with the above constructor
File(FileName file, ID3v2::FrameFactory *frameFactory,
bool readProperties = true,
Properties::ReadStyle propertiesStyle = Properties::Average);
/*! /*!
* Constructs a FLAC file from \a stream. If \a readProperties is true the * Returns a pointer to the ID3v2 tag of the file.
* file's audio properties will also be read. *
* * If \a create is false (the default) this returns a null pointer
* \note TagLib will *not* take ownership of the stream, the caller is * if there is no valid ID3v2 tag.
* responsible for deleting it after the File object. * If \a create is true it will create an ID3v2 tag if one does not exist and returns a valid pointer.
* *
* If this file contains and ID3v2 tag the frames will be created using * \note This may return a valid pointer regardless of whether or not the file on disk has an ID3v2 tag.
* \a frameFactory. * Use hasID3v2Tag() to check if the file on disk actually has an ID3v2 tag.
* *
* \note In the current implementation, \a propertiesStyle is ignored. * \note The Tag <b>is still</b> owned by the MPEG::File and should not be
*/ * deleted by the user. It will be deleted when the file (object) is destroyed.
// BIC: merge with the above constructor *
File(IOStream *stream, ID3v2::FrameFactory *frameFactory, * \see hasID3v2Tag()
bool readProperties = true, */
Properties::ReadStyle propertiesStyle = Properties::Average); ID3v2::Tag *ID3v2Tag(bool create = false);
/*! /*!
* Destroys this instance of the File. * Returns a pointer to the ID3v1 tag of the file.
*/ *
virtual ~File(); * If \a create is false (the default) this returns a null pointer if there is no valid APE tag.
* If \a create is true it will create an APE tag if one does not exist and returns a valid pointer.
*
* \note This may return a valid pointer regardless of whether or not the file on disk has an ID3v1 tag.
* Use hasID3v1Tag() to check if the file on disk actually has an ID3v1 tag.
*
* \note The Tag <b>is still</b> owned by the MPEG::File and should not be deleted by the user.
* It will be deleted when the file (object) is destroyed.
*
* \see hasID3v1Tag()
*/
ID3v1::Tag *ID3v1Tag(bool create = false);
/*! /*!
* Returns the Tag for this file. This will be a union of XiphComment, * Returns a pointer to the XiphComment for the file.
* ID3v1 and ID3v2 tags. *
* * If \a create is false (the default) this returns a null pointer if there is no valid XiphComment.
* \see ID3v2Tag() * If \a create is true it will create a XiphComment if one does not exist and returns a valid pointer.
* \see ID3v1Tag() *
* \see XiphComment() * \note This may return a valid pointer regardless of whether or not the file on disk has a XiphComment.
*/ * Use hasXiphComment() to check if the file on disk actually has a XiphComment.
virtual Strawberry_TagLib::TagLib::Tag *tag() const; *
* \note The Tag <b>is still</b> owned by the FLAC::File and should not be deleted by the user.
* It will be deleted when the file (object) is destroyed.
*
* \see hasXiphComment()
*/
Ogg::XiphComment *xiphComment(bool create = false);
/*! /*!
* Implements the unified property interface -- export function. * Returns a list of pictures attached to the FLAC file.
* If the file contains more than one tag (e.g. XiphComment and ID3v1), */
* only the first one (in the order XiphComment, ID3v2, ID3v1) will be List<Picture *> pictureList();
* converted to the PropertyMap.
*/
PropertyMap properties() const;
void removeUnsupportedProperties(const StringList &); /*!
* Removes an attached picture. The picture's memory will be freed.
*/
void removePicture(Picture *picture);
/*! /*!
* Implements the unified property interface -- import function. * Remove all attached images.
* This always creates a Xiph comment, if none exists. The return value */
* relates to the Xiph comment only. void removePictures();
* Ignores any changes to ID3v1 or ID3v2 comments since they are not allowed
* in the FLAC specification.
*/
PropertyMap setProperties(const PropertyMap &);
/*! /*!
* Returns the FLAC::Properties for this file. If no audio properties * Add a new picture to the file.
* were read then this will return a null pointer. * The file takes ownership of the picture and will handle freeing its memory.
*/ *
virtual Properties *audioProperties() const; * \note The file will be saved only after calling save().
*/
void addPicture(Picture *picture);
/*! /*!
* Save the file. This will primarily save the XiphComment, but * This will remove the tags that match the OR-ed together TagTypes from the file.
* will also keep any old ID3-tags up to date. If the file * By default it removes all tags.
* has no XiphComment, one will be constructed from the ID3-tags. *
* * \warning This will also invalidate pointers to the tags as their memory will be freed.
* This returns true if the save was successful. *
*/ * \note In order to make the removal permanent save() still needs to be called.
virtual bool save(); *
* \note This won't remove the Vorbis comment block completely.
* The vendor ID will be preserved.
*/
void strip(int tags = AllTags);
/*! /*!
* Returns a pointer to the ID3v2 tag of the file. * Returns whether or not the file on disk actually has a XiphComment.
* *
* If \a create is false (the default) this returns a null pointer * \see xiphComment()
* if there is no valid ID3v2 tag. If \a create is true it will create */
* an ID3v2 tag if one does not exist and returns a valid pointer. bool hasXiphComment() const;
*
* \note This may return a valid pointer regardless of whether or not the
* file on disk has an ID3v2 tag. Use hasID3v2Tag() to check if the file
* on disk actually has an ID3v2 tag.
*
* \note The Tag <b>is still</b> owned by the MPEG::File and should not be
* deleted by the user. It will be deleted when the file (object) is
* destroyed.
*
* \see hasID3v2Tag()
*/
ID3v2::Tag *ID3v2Tag(bool create = false);
/*! /*!
* Returns a pointer to the ID3v1 tag of the file. * Returns whether or not the file on disk actually has an ID3v1 tag.
* *
* If \a create is false (the default) this returns a null pointer * \see ID3v1Tag()
* if there is no valid APE tag. If \a create is true it will create */
* an APE tag if one does not exist and returns a valid pointer. bool hasID3v1Tag() const;
*
* \note This may return a valid pointer regardless of whether or not the
* file on disk has an ID3v1 tag. Use hasID3v1Tag() to check if the file
* on disk actually has an ID3v1 tag.
*
* \note The Tag <b>is still</b> owned by the MPEG::File and should not be
* deleted by the user. It will be deleted when the file (object) is
* destroyed.
*
* \see hasID3v1Tag()
*/
ID3v1::Tag *ID3v1Tag(bool create = false);
/*! /*!
* Returns a pointer to the XiphComment for the file. * Returns whether or not the file on disk actually has an ID3v2 tag.
* *
* If \a create is false (the default) this returns a null pointer * \see ID3v2Tag()
* if there is no valid XiphComment. If \a create is true it will create */
* a XiphComment if one does not exist and returns a valid pointer. bool hasID3v2Tag() const;
*
* \note This may return a valid pointer regardless of whether or not the
* file on disk has a XiphComment. Use hasXiphComment() to check if the
* file on disk actually has a XiphComment.
*
* \note The Tag <b>is still</b> owned by the FLAC::File and should not be
* deleted by the user. It will be deleted when the file (object) is
* destroyed.
*
* \see hasXiphComment()
*/
Ogg::XiphComment *xiphComment(bool create = false);
/*! /*!
* Set the ID3v2::FrameFactory to something other than the default. This * Returns whether or not the given \a stream can be opened as a FLAC file.
* can be used to specify the way that ID3v2 frames will be interpreted *
* when * \note This method is designed to do a quick check.
* * The result may not necessarily be correct.
* \see ID3v2FrameFactory */
* \deprecated This value should be passed in via the constructor static bool isSupported(IOStream *stream);
*/
void setID3v2FrameFactory(const ID3v2::FrameFactory *factory);
/*! private:
* Returns the block of data used by FLAC::Properties for parsing the File(const File&);
* stream properties. File &operator=(const File&);
*
* \deprecated Always returns an empty vector.
*/
ByteVector streamInfoData(); // BIC: remove
/*! void read(bool readProperties);
* Returns the length of the audio-stream, used by FLAC::Properties for void scan();
* calculating the bitrate.
*
* \deprecated Always returns zero.
*/
long streamLength(); // BIC: remove
/*! class FilePrivate;
* Returns a list of pictures attached to the FLAC file. FilePrivate *d;
*/ };
List<Picture *> pictureList(); } // namespace FLAC
} // namespace TagLib
/*! } // namespace Strawberry_TagLib
* Removes an attached picture. If \a del is true the picture's memory
* will be freed; if it is false, it must be deleted by the user.
*/
void removePicture(Picture *picture, bool del = true);
/*!
* Remove all attached images.
*/
void removePictures();
/*!
* Add a new picture to the file. The file takes ownership of the
* picture and will handle freeing its memory.
*
* \note The file will be saved only after calling save().
*/
void addPicture(Picture *picture);
/*!
* This will remove the tags that match the OR-ed together TagTypes from
* the file. By default it removes all tags.
*
* \warning This will also invalidate pointers to the tags as their memory
* will be freed.
*
* \note In order to make the removal permanent save() still needs to be
* called.
*
* \note This won't remove the Vorbis comment block completely. The
* vendor ID will be preserved.
*/
void strip(int tags = AllTags);
/*!
* Returns whether or not the file on disk actually has a XiphComment.
*
* \see xiphComment()
*/
bool hasXiphComment() const;
/*!
* Returns whether or not the file on disk actually has an ID3v1 tag.
*
* \see ID3v1Tag()
*/
bool hasID3v1Tag() const;
/*!
* Returns whether or not the file on disk actually has an ID3v2 tag.
*
* \see ID3v2Tag()
*/
bool hasID3v2Tag() const;
/*!
* Returns whether or not the given \a stream can be opened as a FLAC
* file.
*
* \note This method is designed to do a quick check. The result may
* not necessarily be correct.
*/
static bool isSupported(IOStream *stream);
private:
File(const File &);
File &operator=(const File &);
void read(bool readProperties);
void scan();
class FilePrivate;
FilePrivate *d;
};
}
}
}
#endif #endif

View File

@@ -23,25 +23,17 @@
* http://www.mozilla.org/MPL/ * * http://www.mozilla.org/MPL/ *
***************************************************************************/ ***************************************************************************/
#include <taglib.h> #include "taglib.h"
#include <tdebug.h> #include "tdebug.h"
#include "flacmetadatablock.h" #include "flacmetadatablock.h"
using namespace Strawberry_TagLib::TagLib; using namespace Strawberry_TagLib::TagLib;
class FLAC::MetadataBlock::MetadataBlockPrivate class FLAC::MetadataBlock::MetadataBlockPrivate {
{ public:
public:
MetadataBlockPrivate() {} MetadataBlockPrivate() {}
}; };
FLAC::MetadataBlock::MetadataBlock() FLAC::MetadataBlock::MetadataBlock() : d(nullptr) {}
{
d = 0;
}
FLAC::MetadataBlock::~MetadataBlock()
{
}
FLAC::MetadataBlock::~MetadataBlock() {}

View File

@@ -32,46 +32,43 @@
namespace Strawberry_TagLib { namespace Strawberry_TagLib {
namespace TagLib { namespace TagLib {
namespace FLAC {
namespace FLAC { class TAGLIB_EXPORT MetadataBlock {
public:
explicit MetadataBlock();
virtual ~MetadataBlock();
class TAGLIB_EXPORT MetadataBlock enum BlockType {
{ StreamInfo = 0,
public: Padding,
MetadataBlock(); Application,
virtual ~MetadataBlock(); SeekTable,
VorbisComment,
CueSheet,
Picture
};
enum BlockType { /*!
StreamInfo = 0, * Returns the FLAC metadata block type.
Padding, */
Application, virtual int code() const = 0;
SeekTable,
VorbisComment,
CueSheet,
Picture
};
/*! /*!
* Returns the FLAC metadata block type. * Render the content of the block.
*/ */
virtual int code() const = 0; virtual ByteVector render() const = 0;
/*! private:
* Render the content of the block. MetadataBlock(const MetadataBlock &item);
*/ MetadataBlock &operator=(const MetadataBlock &item);
virtual ByteVector render() const = 0;
private: class MetadataBlockPrivate;
MetadataBlock(const MetadataBlock &item); MetadataBlockPrivate *d;
MetadataBlock &operator=(const MetadataBlock &item); };
class MetadataBlockPrivate; } // namespace FLAC
MetadataBlockPrivate *d; } // namespace TagLib
}; } // namespace Strawberry_TagLib
}
}
}
#endif #endif

View File

@@ -23,22 +23,20 @@
* http://www.mozilla.org/MPL/ * * http://www.mozilla.org/MPL/ *
***************************************************************************/ ***************************************************************************/
#include <taglib.h> #include "taglib.h"
#include <tdebug.h> #include "tdebug.h"
#include "flacpicture.h" #include "flacpicture.h"
using namespace Strawberry_TagLib::TagLib; using namespace Strawberry_TagLib::TagLib;
class FLAC::Picture::PicturePrivate class FLAC::Picture::PicturePrivate {
{ public:
public: explicit PicturePrivate() : type(FLAC::Picture::Other),
PicturePrivate() : width(0),
type(FLAC::Picture::Other), height(0),
width(0), colorDepth(0),
height(0), numColors(0) {
colorDepth(0), }
numColors(0)
{}
Type type; Type type;
String mimeType; String mimeType;
@@ -50,168 +48,147 @@ public:
ByteVector data; ByteVector data;
}; };
FLAC::Picture::Picture() : FLAC::Picture::Picture() : d(new PicturePrivate()) {
d(new PicturePrivate())
{
} }
FLAC::Picture::Picture(const ByteVector &data) : FLAC::Picture::Picture(const ByteVector &data) : d(new PicturePrivate()) {
d(new PicturePrivate())
{
parse(data); parse(data);
} }
FLAC::Picture::~Picture() FLAC::Picture::~Picture() {
{
delete d; delete d;
} }
int FLAC::Picture::code() const int FLAC::Picture::code() const {
{
return FLAC::MetadataBlock::Picture; return FLAC::MetadataBlock::Picture;
} }
bool FLAC::Picture::parse(const ByteVector &data) bool FLAC::Picture::parse(const ByteVector &data) {
{
if(data.size() < 32) { if (data.size() < 32) {
debug("A picture block must contain at least 5 bytes."); debug("A picture block must contain at least 5 bytes.");
return false; return false;
} }
unsigned int pos = 0; size_t pos = 0;
d->type = FLAC::Picture::Type(data.toUInt(pos)); d->type = FLAC::Picture::Type(data.toUInt32BE(pos));
pos += 4; pos += 4;
unsigned int mimeTypeLength = data.toUInt(pos); const unsigned int mimeTypeLength = data.toUInt32BE(pos);
pos += 4; pos += 4;
if(pos + mimeTypeLength + 24 > data.size()) { if (pos + mimeTypeLength + 24 > data.size()) {
debug("Invalid picture block."); debug("Invalid picture block.");
return false; return false;
} }
d->mimeType = String(data.mid(pos, mimeTypeLength), String::UTF8); d->mimeType = String(data.mid(pos, mimeTypeLength), String::UTF8);
pos += mimeTypeLength; pos += mimeTypeLength;
unsigned int descriptionLength = data.toUInt(pos); const unsigned int descriptionLength = data.toUInt32BE(pos);
pos += 4; pos += 4;
if(pos + descriptionLength + 20 > data.size()) { if (pos + descriptionLength + 20 > data.size()) {
debug("Invalid picture block."); debug("Invalid picture block.");
return false; return false;
} }
d->description = String(data.mid(pos, descriptionLength), String::UTF8); d->description = String(data.mid(pos, descriptionLength), String::UTF8);
pos += descriptionLength; pos += descriptionLength;
d->width = data.toUInt(pos); d->width = data.toUInt32BE(pos);
pos += 4; pos += 4;
d->height = data.toUInt(pos); d->height = data.toUInt32BE(pos);
pos += 4; pos += 4;
d->colorDepth = data.toUInt(pos); d->colorDepth = data.toUInt32BE(pos);
pos += 4; pos += 4;
d->numColors = data.toUInt(pos); d->numColors = data.toUInt32BE(pos);
pos += 4; pos += 4;
unsigned int dataLength = data.toUInt(pos); const unsigned int dataLength = data.toUInt32BE(pos);
pos += 4; pos += 4;
if(pos + dataLength > data.size()) { if (pos + dataLength > data.size()) {
debug("Invalid picture block."); debug("Invalid picture block.");
return false; return false;
} }
d->data = data.mid(pos, dataLength); d->data = data.mid(pos, dataLength);
return true; return true;
} }
ByteVector FLAC::Picture::render() const ByteVector FLAC::Picture::render() const {
{
ByteVector result; ByteVector result;
result.append(ByteVector::fromUInt(d->type)); result.append(ByteVector::fromUInt32BE(d->type));
ByteVector mimeTypeData = d->mimeType.data(String::UTF8); ByteVector mimeTypeData = d->mimeType.data(String::UTF8);
result.append(ByteVector::fromUInt(mimeTypeData.size())); result.append(ByteVector::fromUInt32BE(mimeTypeData.size()));
result.append(mimeTypeData); result.append(mimeTypeData);
ByteVector descriptionData = d->description.data(String::UTF8); ByteVector descriptionData = d->description.data(String::UTF8);
result.append(ByteVector::fromUInt(descriptionData.size())); result.append(ByteVector::fromUInt32BE(descriptionData.size()));
result.append(descriptionData); result.append(descriptionData);
result.append(ByteVector::fromUInt(d->width)); result.append(ByteVector::fromUInt32BE(d->width));
result.append(ByteVector::fromUInt(d->height)); result.append(ByteVector::fromUInt32BE(d->height));
result.append(ByteVector::fromUInt(d->colorDepth)); result.append(ByteVector::fromUInt32BE(d->colorDepth));
result.append(ByteVector::fromUInt(d->numColors)); result.append(ByteVector::fromUInt32BE(d->numColors));
result.append(ByteVector::fromUInt(d->data.size())); result.append(ByteVector::fromUInt32BE(d->data.size()));
result.append(d->data); result.append(d->data);
return result; return result;
} }
FLAC::Picture::Type FLAC::Picture::type() const FLAC::Picture::Type FLAC::Picture::type() const {
{
return d->type; return d->type;
} }
void FLAC::Picture::setType(FLAC::Picture::Type type) void FLAC::Picture::setType(FLAC::Picture::Type type) {
{
d->type = type; d->type = type;
} }
String FLAC::Picture::mimeType() const String FLAC::Picture::mimeType() const {
{
return d->mimeType; return d->mimeType;
} }
void FLAC::Picture::setMimeType(const String &mimeType) void FLAC::Picture::setMimeType(const String &mimeType) {
{
d->mimeType = mimeType; d->mimeType = mimeType;
} }
String FLAC::Picture::description() const String FLAC::Picture::description() const {
{
return d->description; return d->description;
} }
void FLAC::Picture::setDescription(const String &description) void FLAC::Picture::setDescription(const String &description) {
{
d->description = description; d->description = description;
} }
int FLAC::Picture::width() const int FLAC::Picture::width() const {
{
return d->width; return d->width;
} }
void FLAC::Picture::setWidth(int width) void FLAC::Picture::setWidth(int width) {
{
d->width = width; d->width = width;
} }
int FLAC::Picture::height() const int FLAC::Picture::height() const {
{
return d->height; return d->height;
} }
void FLAC::Picture::setHeight(int height) void FLAC::Picture::setHeight(int height) {
{
d->height = height; d->height = height;
} }
int FLAC::Picture::colorDepth() const int FLAC::Picture::colorDepth() const {
{
return d->colorDepth; return d->colorDepth;
} }
void FLAC::Picture::setColorDepth(int colorDepth) void FLAC::Picture::setColorDepth(int colorDepth) {
{
d->colorDepth = colorDepth; d->colorDepth = colorDepth;
} }
int FLAC::Picture::numColors() const int FLAC::Picture::numColors() const {
{
return d->numColors; return d->numColors;
} }
void FLAC::Picture::setNumColors(int numColors) void FLAC::Picture::setNumColors(int numColors) {
{
d->numColors = numColors; d->numColors = numColors;
} }
ByteVector FLAC::Picture::data() const ByteVector FLAC::Picture::data() const {
{
return d->data; return d->data;
} }
void FLAC::Picture::setData(const ByteVector &data) void FLAC::Picture::setData(const ByteVector &data) {
{
d->data = data; d->data = data;
} }

View File

@@ -35,176 +35,169 @@
namespace Strawberry_TagLib { namespace Strawberry_TagLib {
namespace TagLib { namespace TagLib {
namespace FLAC { namespace FLAC {
class TAGLIB_EXPORT Picture : public MetadataBlock class TAGLIB_EXPORT Picture : public MetadataBlock {
{ public:
public: /*!
* This describes the function or content of the picture.
*/
enum Type {
//! A type not enumerated below
Other = 0x00,
//! 32x32 PNG image that should be used as the file icon
FileIcon = 0x01,
//! File icon of a different size or format
OtherFileIcon = 0x02,
//! Front cover image of the album
FrontCover = 0x03,
//! Back cover image of the album
BackCover = 0x04,
//! Inside leaflet page of the album
LeafletPage = 0x05,
//! Image from the album itself
Media = 0x06,
//! Picture of the lead artist or soloist
LeadArtist = 0x07,
//! Picture of the artist or performer
Artist = 0x08,
//! Picture of the conductor
Conductor = 0x09,
//! Picture of the band or orchestra
Band = 0x0A,
//! Picture of the composer
Composer = 0x0B,
//! Picture of the lyricist or text writer
Lyricist = 0x0C,
//! Picture of the recording location or studio
RecordingLocation = 0x0D,
//! Picture of the artists during recording
DuringRecording = 0x0E,
//! Picture of the artists during performance
DuringPerformance = 0x0F,
//! Picture from a movie or video related to the track
MovieScreenCapture = 0x10,
//! Picture of a large, coloured fish
ColouredFish = 0x11,
//! Illustration related to the track
Illustration = 0x12,
//! Logo of the band or performer
BandLogo = 0x13,
//! Logo of the publisher (record company)
PublisherLogo = 0x14
};
/*! explicit Picture();
* This describes the function or content of the picture. explicit Picture(const ByteVector &data);
*/ ~Picture() override;
enum Type {
//! A type not enumerated below
Other = 0x00,
//! 32x32 PNG image that should be used as the file icon
FileIcon = 0x01,
//! File icon of a different size or format
OtherFileIcon = 0x02,
//! Front cover image of the album
FrontCover = 0x03,
//! Back cover image of the album
BackCover = 0x04,
//! Inside leaflet page of the album
LeafletPage = 0x05,
//! Image from the album itself
Media = 0x06,
//! Picture of the lead artist or soloist
LeadArtist = 0x07,
//! Picture of the artist or performer
Artist = 0x08,
//! Picture of the conductor
Conductor = 0x09,
//! Picture of the band or orchestra
Band = 0x0A,
//! Picture of the composer
Composer = 0x0B,
//! Picture of the lyricist or text writer
Lyricist = 0x0C,
//! Picture of the recording location or studio
RecordingLocation = 0x0D,
//! Picture of the artists during recording
DuringRecording = 0x0E,
//! Picture of the artists during performance
DuringPerformance = 0x0F,
//! Picture from a movie or video related to the track
MovieScreenCapture = 0x10,
//! Picture of a large, coloured fish
ColouredFish = 0x11,
//! Illustration related to the track
Illustration = 0x12,
//! Logo of the band or performer
BandLogo = 0x13,
//! Logo of the publisher (record company)
PublisherLogo = 0x14
};
Picture(); /*!
Picture(const ByteVector &data); * Returns the type of the image.
~Picture(); */
Type type() const;
/*! /*!
* Returns the type of the image. * Sets the type of the image.
*/ */
Type type() const; void setType(Type type);
/*! /*!
* Sets the type of the image. * Returns the mime type of the image. This should in most cases be "image/png" or "image/jpeg".
*/ */
void setType(Type type); String mimeType() const;
/*! /*!
* Returns the mime type of the image. This should in most cases be * Sets the mime type of the image. This should in most cases be "image/png" or "image/jpeg".
* "image/png" or "image/jpeg". */
*/ void setMimeType(const String &m);
String mimeType() const;
/*! /*!
* Sets the mime type of the image. This should in most cases be * Returns a text description of the image.
* "image/png" or "image/jpeg". */
*/
void setMimeType(const String &m);
/*! String description() const;
* Returns a text description of the image.
*/
String description() const; /*!
* Sets a textual description of the image to \a desc.
*/
/*! void setDescription(const String &desc);
* Sets a textual description of the image to \a desc.
*/
void setDescription(const String &desc); /*!
* Returns the width of the image.
*/
int width() const;
/*! /*!
* Returns the width of the image. * Sets the width of the image.
*/ */
int width() const; void setWidth(int w);
/*! /*!
* Sets the width of the image. * Returns the height of the image.
*/ */
void setWidth(int w); int height() const;
/*! /*!
* Returns the height of the image. * Sets the height of the image.
*/ */
int height() const; void setHeight(int h);
/*! /*!
* Sets the height of the image. * Returns the color depth (in bits-per-pixel) of the image.
*/ */
void setHeight(int h); int colorDepth() const;
/*! /*!
* Returns the color depth (in bits-per-pixel) of the image. * Sets the color depth (in bits-per-pixel) of the image.
*/ */
int colorDepth() const; void setColorDepth(int depth);
/*! /*!
* Sets the color depth (in bits-per-pixel) of the image. * Returns the number of colors used on the image..
*/ */
void setColorDepth(int depth); int numColors() const;
/*! /*!
* Returns the number of colors used on the image.. * Sets the number of colors used on the image (for indexed images).
*/ */
int numColors() const; void setNumColors(int numColors);
/*! /*!
* Sets the number of colors used on the image (for indexed images). * Returns the image data.
*/ */
void setNumColors(int numColors); ByteVector data() const;
/*! /*!
* Returns the image data. * Sets the image data.
*/ */
ByteVector data() const; void setData(const ByteVector &data);
/*! /*!
* Sets the image data. * Returns the FLAC metadata block type.
*/ */
void setData(const ByteVector &data); int code() const override;
/*! /*!
* Returns the FLAC metadata block type. * Render the content to the FLAC picture block format.
*/ */
int code() const; ByteVector render() const override;
/*! /*!
* Render the content to the FLAC picture block format. * Parse the picture data in the FLAC picture block format.
*/ */
ByteVector render() const; bool parse(const ByteVector &rawData);
/*! private:
* Parse the picture data in the FLAC picture block format. Picture(const Picture &item);
*/ Picture &operator=(const Picture &item);
bool parse(const ByteVector &rawData);
private: class PicturePrivate;
Picture(const Picture &item); PicturePrivate *d;
Picture &operator=(const Picture &item); };
class PicturePrivate; } // namespace FLAC
PicturePrivate *d; } // namespace TagLib
}; } // namespace Strawberry_TagLib
typedef List<Picture> PictureList;
}
}
}
#endif #endif

View File

@@ -23,24 +23,22 @@
* http://www.mozilla.org/MPL/ * * http://www.mozilla.org/MPL/ *
***************************************************************************/ ***************************************************************************/
#include <tstring.h> #include "tstring.h"
#include <tdebug.h> #include "tdebug.h"
#include "flacproperties.h" #include "flacproperties.h"
#include "flacfile.h" #include "flacfile.h"
using namespace Strawberry_TagLib::TagLib; using namespace Strawberry_TagLib::TagLib;
class FLAC::Properties::PropertiesPrivate class FLAC::AudioProperties::AudioPropertiesPrivate {
{ public:
public: explicit AudioPropertiesPrivate() : length(0),
PropertiesPrivate() : bitrate(0),
length(0), sampleRate(0),
bitrate(0), bitsPerSample(0),
sampleRate(0), channels(0),
bitsPerSample(0), sampleFrames(0) {}
channels(0),
sampleFrames(0) {}
int length; int length;
int bitrate; int bitrate;
@@ -55,72 +53,43 @@ public:
// public members // public members
//////////////////////////////////////////////////////////////////////////////// ////////////////////////////////////////////////////////////////////////////////
FLAC::Properties::Properties(ByteVector data, long streamLength, ReadStyle style) : FLAC::AudioProperties::AudioProperties(const ByteVector &data, long long streamLength, ReadStyle) : Strawberry_TagLib::TagLib::AudioProperties(), d(new AudioPropertiesPrivate()) {
AudioProperties(style),
d(new PropertiesPrivate())
{
read(data, streamLength); read(data, streamLength);
} }
FLAC::Properties::Properties(File *, ReadStyle style) : FLAC::AudioProperties::~AudioProperties() {
AudioProperties(style),
d(new PropertiesPrivate())
{
debug("FLAC::Properties::Properties() - This constructor is no longer used.");
}
FLAC::Properties::~Properties()
{
delete d; delete d;
} }
int FLAC::Properties::length() const int FLAC::AudioProperties::lengthInSeconds() const {
{
return lengthInSeconds();
}
int FLAC::Properties::lengthInSeconds() const
{
return d->length / 1000; return d->length / 1000;
} }
int FLAC::Properties::lengthInMilliseconds() const int FLAC::AudioProperties::lengthInMilliseconds() const {
{
return d->length; return d->length;
} }
int FLAC::Properties::bitrate() const int FLAC::AudioProperties::bitrate() const {
{
return d->bitrate; return d->bitrate;
} }
int FLAC::Properties::sampleRate() const int FLAC::AudioProperties::sampleRate() const {
{
return d->sampleRate; return d->sampleRate;
} }
int FLAC::Properties::bitsPerSample() const int FLAC::AudioProperties::bitsPerSample() const {
{
return d->bitsPerSample; return d->bitsPerSample;
} }
int FLAC::Properties::sampleWidth() const int FLAC::AudioProperties::channels() const {
{
return bitsPerSample();
}
int FLAC::Properties::channels() const
{
return d->channels; return d->channels;
} }
unsigned long long FLAC::Properties::sampleFrames() const unsigned long long FLAC::AudioProperties::sampleFrames() const {
{
return d->sampleFrames; return d->sampleFrames;
} }
ByteVector FLAC::Properties::signature() const ByteVector FLAC::AudioProperties::signature() const {
{
return d->signature; return d->signature;
} }
@@ -128,14 +97,14 @@ ByteVector FLAC::Properties::signature() const
// private members // private members
//////////////////////////////////////////////////////////////////////////////// ////////////////////////////////////////////////////////////////////////////////
void FLAC::Properties::read(const ByteVector &data, long streamLength) void FLAC::AudioProperties::read(const ByteVector &data, long long streamLength) {
{
if(data.size() < 18) { if (data.size() < 18) {
debug("FLAC::Properties::read() - FLAC properties must contain at least 18 bytes."); debug("FLAC::AudioProperties::read() - FLAC properties must contain at least 18 bytes.");
return; return;
} }
unsigned int pos = 0; size_t pos = 0;
// Minimum block size (in samples) // Minimum block size (in samples)
pos += 2; pos += 2;
@@ -149,28 +118,29 @@ void FLAC::Properties::read(const ByteVector &data, long streamLength)
// Maximum frame size (in bytes) // Maximum frame size (in bytes)
pos += 3; pos += 3;
const unsigned int flags = data.toUInt(pos, true); const unsigned int flags = data.toUInt32BE(pos);
pos += 4; pos += 4;
d->sampleRate = flags >> 12; d->sampleRate = flags >> 12;
d->channels = ((flags >> 9) & 7) + 1; d->channels = ((flags >> 9) & 7) + 1;
d->bitsPerSample = ((flags >> 4) & 31) + 1; d->bitsPerSample = ((flags >> 4) & 31) + 1;
// The last 4 bits are the most significant 4 bits for the 36 bit // The last 4 bits are the most significant 4 bits for the 36 bit
// stream length in samples. (Audio files measured in days) // stream length in samples. (Audio files measured in days)
const unsigned long long hi = flags & 0xf; const unsigned long long hi = flags & 0xf;
const unsigned long long lo = data.toUInt(pos, true); const unsigned long long lo = data.toUInt32BE(pos);
pos += 4; pos += 4;
d->sampleFrames = (hi << 32) | lo; d->sampleFrames = (hi << 32) | lo;
if(d->sampleFrames > 0 && d->sampleRate > 0) { if (d->sampleFrames > 0 && d->sampleRate > 0) {
const double length = d->sampleFrames * 1000.0 / d->sampleRate; const double length = d->sampleFrames * 1000.0 / d->sampleRate;
d->length = static_cast<int>(length + 0.5); d->length = static_cast<int>(length + 0.5);
d->bitrate = static_cast<int>(streamLength * 8.0 / length + 0.5); d->bitrate = static_cast<int>(streamLength * 8.0 / length + 0.5);
} }
if(data.size() >= pos + 16) if (data.size() >= pos + 16)
d->signature = data.mid(pos, 16); d->signature = data.mid(pos, 16);
} }

View File

@@ -32,119 +32,80 @@
namespace Strawberry_TagLib { namespace Strawberry_TagLib {
namespace TagLib { namespace TagLib {
namespace FLAC { namespace FLAC {
class File; class File;
//! An implementation of audio property reading for FLAC //! An implementation of audio property reading for FLAC
/*! /*!
* This reads the data from an FLAC stream found in the AudioProperties * This reads the data from an FLAC stream found in the AudioProperties API.
* API. */
*/
class TAGLIB_EXPORT Properties : public AudioProperties class TAGLIB_EXPORT AudioProperties : public Strawberry_TagLib::TagLib::AudioProperties {
{ public:
public: /*!
/*! * Create an instance of FLAC::AudioProperties with the data read from the ByteVector \a data.
* Create an instance of FLAC::Properties with the data read from the */
* ByteVector \a data. explicit AudioProperties(const ByteVector &data, long long streamLength, ReadStyle style = Average);
*/
// BIC: switch to const reference
Properties(ByteVector data, long streamLength, ReadStyle style = Average);
/*! /*!
* Create an instance of FLAC::Properties with the data read from the * Destroys this FLAC::AudioProperties instance.
* FLAC::File \a file. */
*/ ~AudioProperties() override;
// BIC: remove
Properties(File *file, ReadStyle style = Average);
/*! /*!
* Destroys this FLAC::Properties instance. * Returns the length of the file in seconds. The length is rounded down to the nearest whole second.
*/ *
virtual ~Properties(); * \see lengthInMilliseconds()
*/
int lengthInSeconds() const override;
/*! /*!
* Returns the length of the file in seconds. The length is rounded down to * Returns the length of the file in milliseconds.
* the nearest whole second. *
* * \see lengthInSeconds()
* \note This method is just an alias of lengthInSeconds(). */
* int lengthInMilliseconds() const override;
* \deprecated
*/
virtual int length() const;
/*! /*!
* Returns the length of the file in seconds. The length is rounded down to * Returns the average bit rate of the file in kb/s.
* the nearest whole second. */
* int bitrate() const override;
* \see lengthInMilliseconds()
*/
// BIC: make virtual
int lengthInSeconds() const;
/*! /*!
* Returns the length of the file in milliseconds. * Returns the sample rate in Hz.
* */
* \see lengthInSeconds() int sampleRate() const override;
*/
// BIC: make virtual
int lengthInMilliseconds() const;
/*! /*!
* Returns the average bit rate of the file in kb/s. * Returns the number of audio channels.
*/ */
virtual int bitrate() const; int channels() const override;
/*! /*!
* Returns the sample rate in Hz. * Returns the number of bits per audio sample as read from the FLAC identification header.
*/ */
virtual int sampleRate() const; int bitsPerSample() const;
/*! /*!
* Returns the number of audio channels. * Return the number of sample frames.
*/ */
virtual int channels() const; unsigned long long sampleFrames() const;
/*! /*!
* Returns the number of bits per audio sample as read from the FLAC * Returns the MD5 signature of the uncompressed audio stream as read from the stream info header.
* identification header. */
*/ ByteVector signature() const;
int bitsPerSample() const;
/*! private:
* Returns the sample width as read from the FLAC identification void read(const ByteVector &data, long long streamLength);
* header.
*
* \note This method is just an alias of bitsPerSample().
*
* \deprecated
*/
int sampleWidth() const;
/*! class AudioPropertiesPrivate;
* Return the number of sample frames. AudioPropertiesPrivate *d;
*/ };
unsigned long long sampleFrames() const; } // namespace FLAC
} // namespace TagLib
/*! } // namespace Strawberry_TagLib
* Returns the MD5 signature of the uncompressed audio stream as read
* from the stream info header.
*/
ByteVector signature() const;
private:
Properties(const Properties &);
Properties &operator=(const Properties &);
void read(const ByteVector &data, long streamLength);
class PropertiesPrivate;
PropertiesPrivate *d;
};
}
}
}
#endif #endif

View File

@@ -23,56 +23,46 @@
* http://www.mozilla.org/MPL/ * * http://www.mozilla.org/MPL/ *
***************************************************************************/ ***************************************************************************/
#include <taglib.h> #include "taglib.h"
#include <tdebug.h> #include "tdebug.h"
#include <tstring.h> #include "tstring.h"
#include "flacunknownmetadatablock.h" #include "flacunknownmetadatablock.h"
using namespace Strawberry_TagLib::TagLib; using namespace Strawberry_TagLib::TagLib;
class FLAC::UnknownMetadataBlock::UnknownMetadataBlockPrivate class FLAC::UnknownMetadataBlock::UnknownMetadataBlockPrivate {
{ public:
public: explicit UnknownMetadataBlockPrivate() : code(0) {}
UnknownMetadataBlockPrivate() : code(0) {}
int code; int code;
ByteVector data; ByteVector data;
}; };
FLAC::UnknownMetadataBlock::UnknownMetadataBlock(int code, const ByteVector &data) : FLAC::UnknownMetadataBlock::UnknownMetadataBlock(int code, const ByteVector &data) : d(new UnknownMetadataBlockPrivate()) {
d(new UnknownMetadataBlockPrivate())
{
d->code = code; d->code = code;
d->data = data; d->data = data;
} }
FLAC::UnknownMetadataBlock::~UnknownMetadataBlock() FLAC::UnknownMetadataBlock::~UnknownMetadataBlock() {
{
delete d; delete d;
} }
int FLAC::UnknownMetadataBlock::code() const int FLAC::UnknownMetadataBlock::code() const {
{
return d->code; return d->code;
} }
void FLAC::UnknownMetadataBlock::setCode(int code) void FLAC::UnknownMetadataBlock::setCode(int code) {
{
d->code = code; d->code = code;
} }
ByteVector FLAC::UnknownMetadataBlock::data() const ByteVector FLAC::UnknownMetadataBlock::data() const {
{
return d->data; return d->data;
} }
void FLAC::UnknownMetadataBlock::setData(const ByteVector &data) void FLAC::UnknownMetadataBlock::setData(const ByteVector &data) {
{
d->data = data; d->data = data;
} }
ByteVector FLAC::UnknownMetadataBlock::render() const ByteVector FLAC::UnknownMetadataBlock::render() const {
{
return d->data; return d->data;
} }

View File

@@ -33,51 +33,48 @@
namespace Strawberry_TagLib { namespace Strawberry_TagLib {
namespace TagLib { namespace TagLib {
namespace FLAC {
namespace FLAC { class TAGLIB_EXPORT UnknownMetadataBlock : public MetadataBlock {
public:
explicit UnknownMetadataBlock(int code, const ByteVector &data);
~UnknownMetadataBlock() override;
class TAGLIB_EXPORT UnknownMetadataBlock : public MetadataBlock /*!
{ * Returns the FLAC metadata block type.
public: */
UnknownMetadataBlock(int blockType, const ByteVector &data); int code() const override;
~UnknownMetadataBlock();
/*! /*!
* Returns the FLAC metadata block type. * Sets the FLAC metadata block type.
*/ */
int code() const; void setCode(int code);
/*! /*!
* Sets the FLAC metadata block type. * Returns the FLAC metadata block type.
*/ */
void setCode(int code); ByteVector data() const;
/*! /*!
* Returns the FLAC metadata block type. * Sets the FLAC metadata block type.
*/ */
ByteVector data() const; void setData(const ByteVector &data);
/*! /*!
* Sets the FLAC metadata block type. * Render the content of the block.
*/ */
void setData(const ByteVector &data); ByteVector render() const override;
/*! private:
* Render the content of the block. explicit UnknownMetadataBlock(const MetadataBlock &item);
*/ UnknownMetadataBlock &operator=(const MetadataBlock &item);
ByteVector render() const;
private: class UnknownMetadataBlockPrivate;
UnknownMetadataBlock(const MetadataBlock &item); UnknownMetadataBlockPrivate *d;
UnknownMetadataBlock &operator=(const MetadataBlock &item); };
class UnknownMetadataBlockPrivate; } // namespace FLAC
UnknownMetadataBlockPrivate *d; } // namespace TagLib
}; } // namespace Strawberry_TagLib
}
}
}
#endif #endif

View File

@@ -33,65 +33,43 @@
using namespace Strawberry_TagLib::TagLib; using namespace Strawberry_TagLib::TagLib;
using namespace IT; using namespace IT;
class IT::File::FilePrivate class IT::File::FilePrivate {
{ public:
public: explicit FilePrivate(AudioProperties::ReadStyle propertiesStyle) : properties(propertiesStyle) {}
explicit FilePrivate(AudioProperties::ReadStyle propertiesStyle)
: tag(), properties(propertiesStyle)
{
}
Mod::Tag tag; Mod::Tag tag;
IT::Properties properties; IT::AudioProperties properties;
}; };
IT::File::File(FileName file, bool readProperties, IT::File::File(FileName file, bool readProperties, AudioProperties::ReadStyle propertiesStyle) : Mod::FileBase(file), d(new FilePrivate(propertiesStyle)) {
AudioProperties::ReadStyle propertiesStyle) :
Mod::FileBase(file), if (isOpen())
d(new FilePrivate(propertiesStyle))
{
if(isOpen())
read(readProperties); read(readProperties);
} }
IT::File::File(IOStream *stream, bool readProperties, IT::File::File(IOStream *stream, bool readProperties, AudioProperties::ReadStyle propertiesStyle) : Mod::FileBase(stream), d(new FilePrivate(propertiesStyle)) {
AudioProperties::ReadStyle propertiesStyle) :
Mod::FileBase(stream), if (isOpen())
d(new FilePrivate(propertiesStyle))
{
if(isOpen())
read(readProperties); read(readProperties);
} }
IT::File::~File() IT::File::~File() {
{
delete d; delete d;
} }
Mod::Tag *IT::File::tag() const Mod::Tag *IT::File::tag() const {
{
return &d->tag; return &d->tag;
} }
PropertyMap IT::File::properties() const IT::AudioProperties *IT::File::audioProperties() const {
{
return d->tag.properties();
}
PropertyMap IT::File::setProperties(const PropertyMap &properties)
{
return d->tag.setProperties(properties);
}
IT::Properties *IT::File::audioProperties() const
{
return &d->properties; return &d->properties;
} }
bool IT::File::save() bool IT::File::save() {
{
if(readOnly()) if (readOnly()) {
{
debug("IT::File::save() - Cannot save to a read only file."); debug("IT::File::save() - Cannot save to a read only file.");
return false; return false;
} }
@@ -105,37 +83,37 @@ bool IT::File::save()
unsigned short instrumentCount = 0; unsigned short instrumentCount = 0;
unsigned short sampleCount = 0; unsigned short sampleCount = 0;
if(!readU16L(length) || !readU16L(instrumentCount) || !readU16L(sampleCount)) if (!readU16L(length) || !readU16L(instrumentCount) || !readU16L(sampleCount))
return false; return false;
seek(15, Current); seek(15, Current);
// write comment as instrument and sample names: // write comment as instrument and sample names:
StringList lines = d->tag.comment().split("\n"); StringList lines = d->tag.comment().split("\n");
for(unsigned short i = 0; i < instrumentCount; ++ i) { for (unsigned short i = 0; i < instrumentCount; ++i) {
seek(192L + length + ((long)i << 2)); seek(192L + length + (static_cast<long>(i) << 2));
unsigned long instrumentOffset = 0; unsigned int instrumentOffset = 0;
if(!readU32L(instrumentOffset)) if (!readU32L(instrumentOffset))
return false; return false;
seek(instrumentOffset + 32); seek(instrumentOffset + 32);
if(i < lines.size()) if (i < lines.size())
writeString(lines[i], 25); writeString(lines[i], 25);
else else
writeString(String(), 25); writeString(String(), 25);
writeByte(0); writeByte(0);
} }
for(unsigned short i = 0; i < sampleCount; ++ i) { for (unsigned short i = 0; i < sampleCount; ++i) {
seek(192L + length + ((long)instrumentCount << 2) + ((long)i << 2)); seek(192L + length + (static_cast<long>(instrumentCount) << 2) + (static_cast<long>(i) << 2));
unsigned long sampleOffset = 0; unsigned int sampleOffset = 0;
if(!readU32L(sampleOffset)) if (!readU32L(sampleOffset))
return false; return false;
seek(sampleOffset + 20); seek(sampleOffset + 20);
if((unsigned int)(i + instrumentCount) < lines.size()) if (static_cast<unsigned int>(i + instrumentCount) < lines.size())
writeString(lines[i + instrumentCount], 25); writeString(lines[i + instrumentCount], 25);
else else
writeString(String(), 25); writeString(String(), 25);
@@ -144,41 +122,40 @@ bool IT::File::save()
// write rest as message: // write rest as message:
StringList messageLines; StringList messageLines;
for(unsigned int i = instrumentCount + sampleCount; i < lines.size(); ++ i) for (unsigned int i = instrumentCount + sampleCount; i < lines.size(); ++i)
messageLines.append(lines[i]); messageLines.append(lines[i]);
ByteVector message = messageLines.toString("\r").data(String::Latin1); ByteVector message = messageLines.toString("\r").data(String::Latin1);
// it's actually not really stated if the message needs a // it's actually not really stated if the message needs a
// terminating NUL but it does not hurt to add one: // terminating NUL but it does not hurt to add one:
if(message.size() > 7999) if (message.size() > 7999)
message.resize(7999); message.resize(7999);
message.append((char)0); message.append(static_cast<char>(0));
unsigned short special = 0; unsigned short special = 0;
unsigned short messageLength = 0; unsigned short messageLength = 0;
unsigned long messageOffset = 0; unsigned int messageOffset = 0;
seek(46); seek(46);
if(!readU16L(special)) if (!readU16L(special))
return false; return false;
unsigned long fileSize = File::length(); unsigned int fileSize = File::length();
if(special & Properties::MessageAttached) { if (special & AudioProperties::MessageAttached) {
seek(54); seek(54);
if(!readU16L(messageLength) || !readU32L(messageOffset)) if (!readU16L(messageLength) || !readU32L(messageOffset))
return false; return false;
if(messageLength == 0) if (messageLength == 0)
messageOffset = fileSize; messageOffset = fileSize;
} }
else else {
{
messageOffset = fileSize; messageOffset = fileSize;
seek(46); seek(46);
writeU16L(special | 0x1); writeU16L(special | 0x1);
} }
if(messageOffset + messageLength >= fileSize) { if (messageOffset + messageLength >= fileSize) {
// append new message // append new message
seek(54); seek(54);
writeU16L(message.size()); writeU16L(message.size());
@@ -199,9 +176,8 @@ bool IT::File::save()
return true; return true;
} }
void IT::File::read(bool) void IT::File::read(bool) {
{ if (!isOpen())
if(!isOpen())
return; return;
seek(0); seek(0);
@@ -233,14 +209,14 @@ void IT::File::read(bool)
// sample/instrument names are abused as comments so // sample/instrument names are abused as comments so
// I just add all together. // I just add all together.
String message; String message;
if(special & Properties::MessageAttached) { if (special & AudioProperties::MessageAttached) {
READ_U16L_AS(messageLength); READ_U16L_AS(messageLength);
READ_U32L_AS(messageOffset); READ_U32L_AS(messageOffset);
seek(messageOffset); seek(messageOffset);
ByteVector messageBytes = readBlock(messageLength); ByteVector messageBytes = readBlock(messageLength);
READ_ASSERT(messageBytes.size() == messageLength); READ_ASSERT(messageBytes.size() == messageLength);
int index = messageBytes.find((char) 0); const size_t index = messageBytes.find(static_cast<char>(0));
if(index > -1) if (index != ByteVector::npos())
messageBytes.resize(index, 0); messageBytes.resize(index, 0);
messageBytes.replace('\r', '\n'); messageBytes.replace('\r', '\n');
message = messageBytes; message = messageBytes;
@@ -249,26 +225,26 @@ void IT::File::read(bool)
seek(64); seek(64);
ByteVector pannings = readBlock(64); ByteVector pannings = readBlock(64);
ByteVector volumes = readBlock(64); ByteVector volumes = readBlock(64);
READ_ASSERT(pannings.size() == 64 && volumes.size() == 64); READ_ASSERT(pannings.size() == 64 && volumes.size() == 64);
int channels = 0; int channels = 0;
for(int i = 0; i < 64; ++ i) { for (int i = 0; i < 64; ++i) {
// Strictly speaking an IT file has always 64 channels, but // Strictly speaking an IT file has always 64 channels, but
// I don't count disabled and muted channels. // I don't count disabled and muted channels.
// But this always gives 64 channels for all my files anyway. // But this always gives 64 channels for all my files anyway.
// Strangely VLC does report other values. I wonder how VLC // Strangely VLC does report other values. I wonder how VLC
// gets it's values. // gets it's values.
if((unsigned char) pannings[i] < 128 && volumes[i] > 0) if (static_cast<unsigned char>(pannings[i]) < 128 && volumes[i] > 0)
++channels; ++channels;
} }
d->properties.setChannels(channels); d->properties.setChannels(channels);
// real length might be shorter because of skips and terminator // real length might be shorter because of skips and terminator
unsigned short realLength = 0; unsigned short realLength = 0;
for(unsigned short i = 0; i < length; ++ i) { for (unsigned short i = 0; i < length; ++i) {
READ_BYTE_AS(order); READ_BYTE_AS(order);
if(order == 255) break; if (order == 255) break;
if(order != 254) ++ realLength; if (order != 254) ++realLength;
} }
d->properties.setLengthInPatterns(realLength); d->properties.setLengthInPatterns(realLength);
@@ -277,10 +253,10 @@ void IT::File::read(bool)
// in the instrument/sample names and more characters // in the instrument/sample names and more characters
// afterwards. The spec does not mention such a case. // afterwards. The spec does not mention such a case.
// Currently I just discard anything after a nil, but // Currently I just discard anything after a nil, but
// e.g. VLC seems to interprete a nil as a space. I // e.g. VLC seems to interpret a nil as a space. I
// don't know what is the proper behaviour. // don't know what is the proper behaviour.
for(unsigned short i = 0; i < instrumentCount; ++ i) { for (unsigned short i = 0; i < instrumentCount; ++i) {
seek(192L + length + ((long)i << 2)); seek(192L + length + (static_cast<long>(i) << 2));
READ_U32L_AS(instrumentOffset); READ_U32L_AS(instrumentOffset);
seek(instrumentOffset); seek(instrumentOffset);
@@ -295,8 +271,8 @@ void IT::File::read(bool)
comment.append(instrumentName); comment.append(instrumentName);
} }
for(unsigned short i = 0; i < sampleCount; ++ i) { for (unsigned short i = 0; i < sampleCount; ++i) {
seek(192L + length + ((long)instrumentCount << 2) + ((long)i << 2)); seek(192L + length + (static_cast<long>(instrumentCount) << 2) + (static_cast<long>(i) << 2));
READ_U32L_AS(sampleOffset); READ_U32L_AS(sampleOffset);
seek(sampleOffset); seek(sampleOffset);
@@ -328,8 +304,9 @@ void IT::File::read(bool)
comment.append(sampleName); comment.append(sampleName);
} }
if(message.size() > 0) if (message.size() > 0)
comment.append(message); comment.append(message);
d->tag.setComment(comment.toString("\n")); d->tag.setComment(comment.toString("\n"));
d->tag.setTrackerName("Impulse Tracker"); d->tag.setTrackerName("Impulse Tracker");
} }

View File

@@ -31,81 +31,61 @@
namespace Strawberry_TagLib { namespace Strawberry_TagLib {
namespace TagLib { namespace TagLib {
namespace IT {
namespace IT { class TAGLIB_EXPORT File : public Mod::FileBase {
public:
/*!
* Constructs a Impulse Tracker file from \a file.
*
* \note In the current implementation, both \a readProperties and \a propertiesStyle are ignored.
* The audio properties are always read.
*/
explicit File(FileName file, bool readProperties = true, AudioProperties::ReadStyle propertiesStyle = AudioProperties::Average);
class TAGLIB_EXPORT File : public Mod::FileBase { /*!
public: * Constructs a Impulse Tracker file from \a stream.
/*! *
* Constructs a Impulse Tracker file from \a file. * \note In the current implementation, both \a readProperties and \a propertiesStyle are ignored.
* * The audio properties are always read.
* \note In the current implementation, both \a readProperties and *
* \a propertiesStyle are ignored. The audio properties are always * \note TagLib will *not* take ownership of the stream, the caller is responsible for deleting it after the File object.
* read. */
*/ explicit File(IOStream *stream, bool readProperties = true, AudioProperties::ReadStyle propertiesStyle = AudioProperties::Average);
File(FileName file, bool readProperties = true,
AudioProperties::ReadStyle propertiesStyle =
AudioProperties::Average);
/*! /*!
* Constructs a Impulse Tracker file from \a stream. * Destroys this instance of the File.
* */
* \note In the current implementation, both \a readProperties and ~File() override;
* \a propertiesStyle are ignored. The audio properties are always
* read.
*
* \note TagLib will *not* take ownership of the stream, the caller is
* responsible for deleting it after the File object.
*/
File(IOStream *stream, bool readProperties = true,
AudioProperties::ReadStyle propertiesStyle =
AudioProperties::Average);
/*! Mod::Tag *tag() const override;
* Destroys this instance of the File.
*/
virtual ~File();
Mod::Tag *tag() const; /*!
* Returns the IT::AudioProperties for this file. If no audio properties
* were read then this will return a null pointer.
*/
IT::AudioProperties *audioProperties() const override;
/*! /*!
* Forwards to Mod::Tag::properties(). * Save the file.
* BIC: will be removed once File::toDict() is made virtual * This is the same as calling save(AllTags);
*/ *
PropertyMap properties() const; * \note Saving Impulse Tracker tags is not supported.
*/
bool save() override;
/*! private:
* Forwards to Mod::Tag::setProperties(). File(const File&);
* BIC: will be removed once File::setProperties() is made virtual File &operator=(const File&);
*/
PropertyMap setProperties(const PropertyMap &);
/*! void read(bool readProperties);
* Returns the IT::Properties for this file. If no audio properties
* were read then this will return a null pointer.
*/
IT::Properties *audioProperties() const;
/*! class FilePrivate;
* Save the file. FilePrivate *d;
* This is the same as calling save(AllTags); };
*
* \note Saving Impulse Tracker tags is not supported.
*/
bool save();
} // namespace IT
private: } // namespace TagLib
File(const File &); } // namespace Strawberry_TagLib
File &operator=(const File &);
void read(bool readProperties);
class FilePrivate;
FilePrivate *d;
};
}
}
}
#endif #endif

View File

@@ -29,29 +29,26 @@
using namespace Strawberry_TagLib::TagLib; using namespace Strawberry_TagLib::TagLib;
using namespace IT; using namespace IT;
class IT::Properties::PropertiesPrivate class IT::AudioProperties::AudioPropertiesPrivate {
{ public:
public: explicit AudioPropertiesPrivate() : channels(0),
PropertiesPrivate() : lengthInPatterns(0),
channels(0), instrumentCount(0),
lengthInPatterns(0), sampleCount(0),
instrumentCount(0), patternCount(0),
sampleCount(0), version(0),
patternCount(0), compatibleVersion(0),
version(0), flags(0),
compatibleVersion(0), special(0),
flags(0), globalVolume(0),
special(0), mixVolume(0),
globalVolume(0), tempo(0),
mixVolume(0), bpmSpeed(0),
tempo(0), panningSeparation(0),
bpmSpeed(0), pitchWheelDepth(0) {
panningSeparation(0),
pitchWheelDepth(0)
{
} }
int channels; int channels;
unsigned short lengthInPatterns; unsigned short lengthInPatterns;
unsigned short instrumentCount; unsigned short instrumentCount;
unsigned short sampleCount; unsigned short sampleCount;
@@ -60,201 +57,160 @@ public:
unsigned short compatibleVersion; unsigned short compatibleVersion;
unsigned short flags; unsigned short flags;
unsigned short special; unsigned short special;
unsigned char globalVolume; unsigned char globalVolume;
unsigned char mixVolume; unsigned char mixVolume;
unsigned char tempo; unsigned char tempo;
unsigned char bpmSpeed; unsigned char bpmSpeed;
unsigned char panningSeparation; unsigned char panningSeparation;
unsigned char pitchWheelDepth; unsigned char pitchWheelDepth;
}; };
IT::Properties::Properties(AudioProperties::ReadStyle propertiesStyle) : IT::AudioProperties::AudioProperties(AudioProperties::ReadStyle) : Strawberry_TagLib::TagLib::AudioProperties(), d(new AudioPropertiesPrivate()) {}
AudioProperties(propertiesStyle),
d(new PropertiesPrivate())
{
}
IT::Properties::~Properties() IT::AudioProperties::~AudioProperties() {
{
delete d; delete d;
} }
int IT::Properties::length() const int IT::AudioProperties::lengthInSeconds() const {
{
return 0; return 0;
} }
int IT::Properties::lengthInSeconds() const int IT::AudioProperties::lengthInMilliseconds() const {
{
return 0; return 0;
} }
int IT::Properties::lengthInMilliseconds() const int IT::AudioProperties::bitrate() const {
{
return 0; return 0;
} }
int IT::Properties::bitrate() const int IT::AudioProperties::sampleRate() const {
{
return 0; return 0;
} }
int IT::Properties::sampleRate() const int IT::AudioProperties::channels() const {
{
return 0;
}
int IT::Properties::channels() const
{
return d->channels; return d->channels;
} }
unsigned short IT::Properties::lengthInPatterns() const unsigned short IT::AudioProperties::lengthInPatterns() const {
{
return d->lengthInPatterns; return d->lengthInPatterns;
} }
bool IT::Properties::stereo() const bool IT::AudioProperties::stereo() const {
{
return d->flags & Stereo; return d->flags & Stereo;
} }
unsigned short IT::Properties::instrumentCount() const unsigned short IT::AudioProperties::instrumentCount() const {
{
return d->instrumentCount; return d->instrumentCount;
} }
unsigned short IT::Properties::sampleCount() const unsigned short IT::AudioProperties::sampleCount() const {
{
return d->sampleCount; return d->sampleCount;
} }
unsigned short IT::Properties::patternCount() const unsigned short IT::AudioProperties::patternCount() const {
{
return d->patternCount; return d->patternCount;
} }
unsigned short IT::Properties::version() const unsigned short IT::AudioProperties::version() const {
{
return d->version; return d->version;
} }
unsigned short IT::Properties::compatibleVersion() const unsigned short IT::AudioProperties::compatibleVersion() const {
{
return d->compatibleVersion; return d->compatibleVersion;
} }
unsigned short IT::Properties::flags() const unsigned short IT::AudioProperties::flags() const {
{
return d->flags; return d->flags;
} }
unsigned short IT::Properties::special() const unsigned short IT::AudioProperties::special() const {
{
return d->special; return d->special;
} }
unsigned char IT::Properties::globalVolume() const unsigned char IT::AudioProperties::globalVolume() const {
{
return d->globalVolume; return d->globalVolume;
} }
unsigned char IT::Properties::mixVolume() const unsigned char IT::AudioProperties::mixVolume() const {
{
return d->mixVolume; return d->mixVolume;
} }
unsigned char IT::Properties::tempo() const unsigned char IT::AudioProperties::tempo() const {
{
return d->tempo; return d->tempo;
} }
unsigned char IT::Properties::bpmSpeed() const unsigned char IT::AudioProperties::bpmSpeed() const {
{
return d->bpmSpeed; return d->bpmSpeed;
} }
unsigned char IT::Properties::panningSeparation() const unsigned char IT::AudioProperties::panningSeparation() const {
{
return d->panningSeparation; return d->panningSeparation;
} }
unsigned char IT::Properties::pitchWheelDepth() const unsigned char IT::AudioProperties::pitchWheelDepth() const {
{
return d->pitchWheelDepth; return d->pitchWheelDepth;
} }
void IT::Properties::setChannels(int channels) ////////////////////////////////////////////////////////////////////////////////
{ // private members
////////////////////////////////////////////////////////////////////////////////
void IT::AudioProperties::setChannels(int channels) {
d->channels = channels; d->channels = channels;
} }
void IT::Properties::setLengthInPatterns(unsigned short lengthInPatterns) void IT::AudioProperties::setLengthInPatterns(unsigned short lengthInPatterns) {
{
d->lengthInPatterns = lengthInPatterns; d->lengthInPatterns = lengthInPatterns;
} }
void IT::Properties::setInstrumentCount(unsigned short instrumentCount) void IT::AudioProperties::setInstrumentCount(unsigned short instrumentCount) {
{
d->instrumentCount = instrumentCount; d->instrumentCount = instrumentCount;
} }
void IT::Properties::setSampleCount(unsigned short sampleCount) void IT::AudioProperties::setSampleCount(unsigned short sampleCount) {
{
d->sampleCount = sampleCount; d->sampleCount = sampleCount;
} }
void IT::Properties::setPatternCount(unsigned short patternCount) void IT::AudioProperties::setPatternCount(unsigned short patternCount) {
{
d->patternCount = patternCount; d->patternCount = patternCount;
} }
void IT::Properties::setFlags(unsigned short flags) void IT::AudioProperties::setFlags(unsigned short flags) {
{
d->flags = flags; d->flags = flags;
} }
void IT::Properties::setSpecial(unsigned short special) void IT::AudioProperties::setSpecial(unsigned short special) {
{
d->special = special; d->special = special;
} }
void IT::Properties::setCompatibleVersion(unsigned short compatibleVersion) void IT::AudioProperties::setCompatibleVersion(unsigned short compatibleVersion) {
{
d->compatibleVersion = compatibleVersion; d->compatibleVersion = compatibleVersion;
} }
void IT::Properties::setVersion(unsigned short version) void IT::AudioProperties::setVersion(unsigned short version) {
{
d->version = version; d->version = version;
} }
void IT::Properties::setGlobalVolume(unsigned char globalVolume) void IT::AudioProperties::setGlobalVolume(unsigned char globalVolume) {
{
d->globalVolume = globalVolume; d->globalVolume = globalVolume;
} }
void IT::Properties::setMixVolume(unsigned char mixVolume) void IT::AudioProperties::setMixVolume(unsigned char mixVolume) {
{
d->mixVolume = mixVolume; d->mixVolume = mixVolume;
} }
void IT::Properties::setTempo(unsigned char tempo) void IT::AudioProperties::setTempo(unsigned char tempo) {
{
d->tempo = tempo; d->tempo = tempo;
} }
void IT::Properties::setBpmSpeed(unsigned char bpmSpeed) void IT::AudioProperties::setBpmSpeed(unsigned char bpmSpeed) {
{
d->bpmSpeed = bpmSpeed; d->bpmSpeed = bpmSpeed;
} }
void IT::Properties::setPanningSeparation(unsigned char panningSeparation) void IT::AudioProperties::setPanningSeparation(unsigned char panningSeparation) {
{
d->panningSeparation = panningSeparation; d->panningSeparation = panningSeparation;
} }
void IT::Properties::setPitchWheelDepth(unsigned char pitchWheelDepth) void IT::AudioProperties::setPitchWheelDepth(unsigned char pitchWheelDepth) {
{
d->pitchWheelDepth = pitchWheelDepth; d->pitchWheelDepth = pitchWheelDepth;
} }

View File

@@ -31,79 +31,77 @@
namespace Strawberry_TagLib { namespace Strawberry_TagLib {
namespace TagLib { namespace TagLib {
namespace IT { namespace IT {
class TAGLIB_EXPORT Properties : public AudioProperties { class TAGLIB_EXPORT AudioProperties : public Strawberry_TagLib::TagLib::AudioProperties {
friend class File; friend class File;
public:
/*! Flag bits. */
enum {
Stereo = 1,
Vol0MixOptimizations = 2,
UseInstruments = 4,
LinearSlides = 8,
OldEffects = 16,
LinkEffects = 32,
UseMidiPitchController = 64,
RequestEmbeddedMidiConf = 128
};
/*! Special bits. */ public:
enum { /*! Flag bits. */
MessageAttached = 1, enum {
MidiConfEmbedded = 8 Stereo = 1,
}; Vol0MixOptimizations = 2,
UseInstruments = 4,
LinearSlides = 8,
OldEffects = 16,
LinkEffects = 32,
UseMidiPitchController = 64,
RequestEmbeddedMidiConf = 128
};
Properties(AudioProperties::ReadStyle propertiesStyle); /*! Special bits. */
virtual ~Properties(); enum {
MessageAttached = 1,
MidiConfEmbedded = 8
};
int length() const; explicit AudioProperties(AudioProperties::ReadStyle);
int lengthInSeconds() const; ~AudioProperties() override;
int lengthInMilliseconds() const;
int bitrate() const;
int sampleRate() const;
int channels() const;
unsigned short lengthInPatterns() const; int lengthInSeconds() const override;
bool stereo() const; int lengthInMilliseconds() const override;
unsigned short instrumentCount() const; int bitrate() const override;
unsigned short sampleCount() const; int sampleRate() const override;
unsigned short patternCount() const; int channels() const override;
unsigned short version() const;
unsigned short compatibleVersion() const;
unsigned short flags() const;
unsigned short special() const;
unsigned char globalVolume() const;
unsigned char mixVolume() const;
unsigned char tempo() const;
unsigned char bpmSpeed() const;
unsigned char panningSeparation() const;
unsigned char pitchWheelDepth() const;
void setChannels(int channels); unsigned short lengthInPatterns() const;
void setLengthInPatterns(unsigned short lengthInPatterns); bool stereo() const;
void setInstrumentCount(unsigned short instrumentCount); unsigned short instrumentCount() const;
void setSampleCount (unsigned short sampleCount); unsigned short sampleCount() const;
void setPatternCount(unsigned short patternCount); unsigned short patternCount() const;
void setVersion (unsigned short version); unsigned short version() const;
void setCompatibleVersion(unsigned short compatibleVersion); unsigned short compatibleVersion() const;
void setFlags (unsigned short flags); unsigned short flags() const;
void setSpecial (unsigned short special); unsigned short special() const;
void setGlobalVolume(unsigned char globalVolume); unsigned char globalVolume() const;
void setMixVolume (unsigned char mixVolume); unsigned char mixVolume() const;
void setTempo (unsigned char tempo); unsigned char tempo() const;
void setBpmSpeed (unsigned char bpmSpeed); unsigned char bpmSpeed() const;
void setPanningSeparation(unsigned char panningSeparation); unsigned char panningSeparation() const;
void setPitchWheelDepth (unsigned char pitchWheelDepth); unsigned char pitchWheelDepth() const;
private: private:
Properties(const Properties&); void setChannels(int channels);
Properties &operator=(const Properties&); void setLengthInPatterns(unsigned short lengthInPatterns);
void setInstrumentCount(unsigned short instrumentCount);
void setSampleCount(unsigned short sampleCount);
void setPatternCount(unsigned short patternCount);
void setVersion(unsigned short version);
void setCompatibleVersion(unsigned short compatibleVersion);
void setFlags(unsigned short flags);
void setSpecial(unsigned short special);
void setGlobalVolume(unsigned char globalVolume);
void setMixVolume(unsigned char mixVolume);
void setTempo(unsigned char tempo);
void setBpmSpeed(unsigned char bpmSpeed);
void setPanningSeparation(unsigned char panningSeparation);
void setPitchWheelDepth(unsigned char pitchWheelDepth);
class PropertiesPrivate; private:
PropertiesPrivate *d; class AudioPropertiesPrivate;
}; AudioPropertiesPrivate *d;
} };
} } // namespace IT
} } // namespace TagLib
} // namespace Strawberry_TagLib
#endif #endif

View File

@@ -33,115 +33,95 @@
using namespace Strawberry_TagLib::TagLib; using namespace Strawberry_TagLib::TagLib;
using namespace Mod; using namespace Mod;
class Mod::File::FilePrivate class Mod::File::FilePrivate {
{ public:
public: explicit FilePrivate(AudioProperties::ReadStyle propertiesStyle) : properties(propertiesStyle) {}
explicit FilePrivate(AudioProperties::ReadStyle propertiesStyle)
: properties(propertiesStyle)
{
}
Mod::Tag tag; Mod::Tag tag;
Mod::Properties properties; Mod::AudioProperties properties;
}; };
Mod::File::File(FileName file, bool readProperties, Mod::File::File(FileName file, bool readProperties, AudioProperties::ReadStyle propertiesStyle) : Mod::FileBase(file), d(new FilePrivate(propertiesStyle)) {
AudioProperties::ReadStyle propertiesStyle) :
Mod::FileBase(file), if (isOpen())
d(new FilePrivate(propertiesStyle))
{
if(isOpen())
read(readProperties); read(readProperties);
} }
Mod::File::File(IOStream *stream, bool readProperties, Mod::File::File(IOStream *stream, bool readProperties, AudioProperties::ReadStyle propertiesStyle) : Mod::FileBase(stream), d(new FilePrivate(propertiesStyle)) {
AudioProperties::ReadStyle propertiesStyle) :
Mod::FileBase(stream), if (isOpen())
d(new FilePrivate(propertiesStyle))
{
if(isOpen())
read(readProperties); read(readProperties);
} }
Mod::File::~File() Mod::File::~File() {
{
delete d; delete d;
} }
Mod::Tag *Mod::File::tag() const Mod::Tag *Mod::File::tag() const {
{
return &d->tag; return &d->tag;
} }
Mod::Properties *Mod::File::audioProperties() const Mod::AudioProperties *Mod::File::audioProperties() const {
{
return &d->properties; return &d->properties;
} }
PropertyMap Mod::File::properties() const bool Mod::File::save() {
{
return d->tag.properties();
}
PropertyMap Mod::File::setProperties(const PropertyMap &properties) if (readOnly()) {
{
return d->tag.setProperties(properties);
}
bool Mod::File::save()
{
if(readOnly()) {
debug("Mod::File::save() - Cannot save to a read only file."); debug("Mod::File::save() - Cannot save to a read only file.");
return false; return false;
} }
seek(0); seek(0);
writeString(d->tag.title(), 20); writeString(d->tag.title(), 20);
StringList lines = d->tag.comment().split("\n"); StringList lines = d->tag.comment().split("\n");
unsigned int n = std::min(lines.size(), d->properties.instrumentCount()); size_t n = std::min<size_t>(lines.size(), d->properties.instrumentCount());
for(unsigned int i = 0; i < n; ++ i) { for (size_t i = 0; i < n; ++i) {
writeString(lines[i], 22); writeString(lines[i], 22);
seek(8, Current); seek(8, Current);
} }
for(unsigned int i = n; i < d->properties.instrumentCount(); ++ i) { for (size_t i = n; i < d->properties.instrumentCount(); ++i) {
writeString(String(), 22); writeString(String(), 22);
seek(8, Current); seek(8, Current);
} }
return true; return true;
} }
void Mod::File::read(bool) void Mod::File::read(bool) {
{
if(!isOpen()) if (!isOpen())
return; return;
seek(1080); seek(1080);
ByteVector modId = readBlock(4); ByteVector modId = readBlock(4);
READ_ASSERT(modId.size() == 4); READ_ASSERT(modId.size() == 4);
int channels = 4; int channels = 4;
unsigned int instruments = 31; unsigned int instruments = 31;
if(modId == "M.K." || modId == "M!K!" || modId == "M&K!" || modId == "N.T.") { if (modId == "M.K." || modId == "M!K!" || modId == "M&K!" || modId == "N.T.") {
d->tag.setTrackerName("ProTracker"); d->tag.setTrackerName("ProTracker");
channels = 4; channels = 4;
} }
else if(modId.startsWith("FLT") || modId.startsWith("TDZ")) { else if (modId.startsWith("FLT") || modId.startsWith("TDZ")) {
d->tag.setTrackerName("StarTrekker"); d->tag.setTrackerName("StarTrekker");
char digit = modId[3]; char digit = modId[3];
READ_ASSERT(digit >= '0' && digit <= '9'); READ_ASSERT(digit >= '0' && digit <= '9');
channels = digit - '0'; channels = digit - '0';
} }
else if(modId.endsWith("CHN")) { else if (modId.endsWith("CHN")) {
d->tag.setTrackerName("StarTrekker"); d->tag.setTrackerName("StarTrekker");
char digit = modId[0]; char digit = modId[0];
READ_ASSERT(digit >= '0' && digit <= '9'); READ_ASSERT(digit >= '0' && digit <= '9');
channels = digit - '0'; channels = digit - '0';
} }
else if(modId == "CD81" || modId == "OKTA") { else if (modId == "CD81" || modId == "OKTA") {
d->tag.setTrackerName("Atari Oktalyzer"); d->tag.setTrackerName("Atari Oktalyzer");
channels = 8; channels = 8;
} }
else if(modId.endsWith("CH") || modId.endsWith("CN")) { else if (modId.endsWith("CH") || modId.endsWith("CN")) {
d->tag.setTrackerName("TakeTracker"); d->tag.setTrackerName("TakeTracker");
char digit = modId[0]; char digit = modId[0];
READ_ASSERT(digit >= '0' && digit <= '9'); READ_ASSERT(digit >= '0' && digit <= '9');
@@ -153,8 +133,8 @@ void Mod::File::read(bool)
else { else {
// Not sure if this is correct. I'd need a file // Not sure if this is correct. I'd need a file
// created with NoiseTracker to check this. // created with NoiseTracker to check this.
d->tag.setTrackerName("NoiseTracker"); // probably d->tag.setTrackerName("NoiseTracker"); // probably
channels = 4; channels = 4;
instruments = 15; instruments = 15;
} }
d->properties.setChannels(channels); d->properties.setChannels(channels);
@@ -164,7 +144,7 @@ void Mod::File::read(bool)
READ_STRING(d->tag.setTitle, 20); READ_STRING(d->tag.setTitle, 20);
StringList comment; StringList comment;
for(unsigned int i = 0; i < instruments; ++ i) { for (unsigned int i = 0; i < instruments; ++i) {
READ_STRING_AS(instrumentName, 22); READ_STRING_AS(instrumentName, 22);
// value in words, * 2 (<< 1) for bytes: // value in words, * 2 (<< 1) for bytes:
READ_U16B_AS(sampleLength); READ_U16B_AS(sampleLength);
@@ -172,10 +152,10 @@ void Mod::File::read(bool)
READ_BYTE_AS(fineTuneByte); READ_BYTE_AS(fineTuneByte);
int fineTune = fineTuneByte & 0xF; int fineTune = fineTuneByte & 0xF;
// > 7 means negative value // > 7 means negative value
if(fineTune > 7) fineTune -= 16; if (fineTune > 7) fineTune -= 16;
READ_BYTE_AS(volume); READ_BYTE_AS(volume);
if(volume > 64) volume = 64; if (volume > 64) volume = 64;
// volume in decibels: 20 * log10(volume / 64) // volume in decibels: 20 * log10(volume / 64)
// value in words, * 2 (<< 1) for bytes: // value in words, * 2 (<< 1) for bytes:
@@ -189,4 +169,5 @@ void Mod::File::read(bool)
READ_BYTE(d->properties.setLengthInPatterns); READ_BYTE(d->properties.setLengthInPatterns);
d->tag.setComment(comment.toString("\n")); d->tag.setComment(comment.toString("\n"));
} }

View File

@@ -35,82 +35,61 @@
namespace Strawberry_TagLib { namespace Strawberry_TagLib {
namespace TagLib { namespace TagLib {
namespace Mod {
namespace Mod { class TAGLIB_EXPORT File : public Strawberry_TagLib::TagLib::Mod::FileBase {
public:
/*!
* Constructs a Protracker file from \a file.
*
* \note In the current implementation, both \a readProperties and \a propertiesStyle are ignored.
* The audio properties are always read.
*/
explicit File(FileName file, bool readProperties = true, AudioProperties::ReadStyle propertiesStyle = AudioProperties::Average);
class TAGLIB_EXPORT File : public Strawberry_TagLib::TagLib::Mod::FileBase /*!
{ * Constructs a Protracker file from \a stream.
public: *
/*! * \note In the current implementation, both \a readProperties and \a propertiesStyle are ignored.
* Constructs a Protracker file from \a file. * The audio properties are always read.
* *
* \note In the current implementation, both \a readProperties and * \note TagLib will *not* take ownership of the stream, the caller is
* \a propertiesStyle are ignored. The audio properties are always * responsible for deleting it after the File object.
* read. */
*/ explicit File(IOStream *stream, bool readProperties = true, AudioProperties::ReadStyle propertiesStyle = AudioProperties::Average);
File(FileName file, bool readProperties = true,
AudioProperties::ReadStyle propertiesStyle =
AudioProperties::Average);
/*! /*!
* Constructs a Protracker file from \a stream. * Destroys this instance of the File.
* */
* \note In the current implementation, both \a readProperties and ~File() override;
* \a propertiesStyle are ignored. The audio properties are always
* read.
*
* \note TagLib will *not* take ownership of the stream, the caller is
* responsible for deleting it after the File object.
*/
File(IOStream *stream, bool readProperties = true,
AudioProperties::ReadStyle propertiesStyle =
AudioProperties::Average);
/*! Mod::Tag *tag() const override;
* Destroys this instance of the File.
*/
virtual ~File();
Mod::Tag *tag() const; /*!
* Returns the Mod::AudioProperties for this file. If no audio properties were read then this will return a null pointer.
*/
Mod::AudioProperties *audioProperties() const override;
/*! /*!
* Implements the unified property interface -- export function. * Save the file.
* Forwards to Mod::Tag::properties(). * This is the same as calling save(AllTags);
*/ *
PropertyMap properties() const; * \note Saving Protracker tags is not supported.
*/
bool save() override;
/*! private:
* Implements the unified property interface -- import function. File(const File &);
* Forwards to Mod::Tag::setProperties(). File &operator=(const File &);
*/
PropertyMap setProperties(const PropertyMap &);
/*!
* Returns the Mod::Properties for this file. If no audio properties
* were read then this will return a null pointer.
*/
Mod::Properties *audioProperties() const;
/*! void read(bool readProperties);
* Save the file.
* This is the same as calling save(AllTags);
*
* \note Saving Protracker tags is not supported.
*/
bool save();
private: class FilePrivate;
File(const File &); FilePrivate *d;
File &operator=(const File &); };
void read(bool readProperties); } // namespace Mod
} // namespace TagLib
class FilePrivate; } // namespace Strawberry_TagLib
FilePrivate *d;
};
}
}
}
#endif #endif

View File

@@ -30,96 +30,93 @@
using namespace Strawberry_TagLib::TagLib; using namespace Strawberry_TagLib::TagLib;
using namespace Mod; using namespace Mod;
Mod::FileBase::FileBase(FileName file) : Strawberry_TagLib::TagLib::File(file) Mod::FileBase::FileBase(FileName file) : Strawberry_TagLib::TagLib::File(file) {}
{
}
Mod::FileBase::FileBase(IOStream *stream) : Strawberry_TagLib::TagLib::File(stream) Mod::FileBase::FileBase(IOStream *stream) : Strawberry_TagLib::TagLib::File(stream) {}
{
}
void Mod::FileBase::writeString(const String &s, unsigned long size, char padding) void Mod::FileBase::writeString(const String &s, unsigned int size, char padding) {
{
ByteVector data(s.data(String::Latin1)); ByteVector data(s.data(String::Latin1));
data.resize(size, padding); data.resize(size, padding);
writeBlock(data); writeBlock(data);
} }
bool Mod::FileBase::readString(String &s, unsigned long size) bool Mod::FileBase::readString(String &s, unsigned int size) {
{
ByteVector data(readBlock(size)); ByteVector data(readBlock(size));
if(data.size() < size) return false; if (data.size() < size) return false;
int index = data.find((char) 0); const size_t index = data.find(static_cast<char>(0));
if(index > -1) if (index != ByteVector::npos()) {
{
data.resize(index); data.resize(index);
} }
data.replace('\xff', ' '); data.replace('\xff', ' ');
s = data; s = data;
return true; return true;
} }
void Mod::FileBase::writeByte(unsigned char byte) void Mod::FileBase::writeByte(unsigned char _byte) {
{ ByteVector data(1, _byte);
ByteVector data(1, byte);
writeBlock(data); writeBlock(data);
} }
void Mod::FileBase::writeU16L(unsigned short number) void Mod::FileBase::writeU16L(unsigned short number) {
{ writeBlock(ByteVector::fromUInt16LE(number));
writeBlock(ByteVector::fromShort(number, false));
} }
void Mod::FileBase::writeU32L(unsigned long number) void Mod::FileBase::writeU32L(unsigned int number) {
{ writeBlock(ByteVector::fromUInt32LE(number));
writeBlock(ByteVector::fromUInt(number, false));
} }
void Mod::FileBase::writeU16B(unsigned short number) void Mod::FileBase::writeU16B(unsigned short number) {
{ writeBlock(ByteVector::fromUInt16BE(number));
writeBlock(ByteVector::fromShort(number, true));
} }
void Mod::FileBase::writeU32B(unsigned long number) void Mod::FileBase::writeU32B(unsigned int number) {
{ writeBlock(ByteVector::fromUInt32BE(number));
writeBlock(ByteVector::fromUInt(number, true));
} }
bool Mod::FileBase::readByte(unsigned char &byte) bool Mod::FileBase::readByte(unsigned char &_byte) {
{
ByteVector data(readBlock(1)); ByteVector data(readBlock(1));
if(data.size() < 1) return false; if (data.size() < 1) return false;
byte = data[0]; _byte = data[0];
return true; return true;
} }
bool Mod::FileBase::readU16L(unsigned short &number) bool Mod::FileBase::readU16L(unsigned short &number) {
{
ByteVector data(readBlock(2)); ByteVector data(readBlock(2));
if(data.size() < 2) return false; if (data.size() < 2) return false;
number = data.toUShort(false); number = data.toUInt16LE(0);
return true; return true;
} }
bool Mod::FileBase::readU32L(unsigned long &number) { bool Mod::FileBase::readU32L(unsigned int &number) {
ByteVector data(readBlock(4)); ByteVector data(readBlock(4));
if(data.size() < 4) return false; if (data.size() < 4) return false;
number = data.toUInt(false); number = data.toUInt32LE(0);
return true; return true;
} }
bool Mod::FileBase::readU16B(unsigned short &number) bool Mod::FileBase::readU16B(unsigned short &number) {
{
ByteVector data(readBlock(2)); ByteVector data(readBlock(2));
if(data.size() < 2) return false; if (data.size() < 2) return false;
number = data.toUShort(true); number = data.toUInt16BE(0);
return true; return true;
} }
bool Mod::FileBase::readU32B(unsigned long &number) { bool Mod::FileBase::readU32B(unsigned int &number) {
ByteVector data(readBlock(4)); ByteVector data(readBlock(4));
if(data.size() < 4) return false; if (data.size() < 4) return false;
number = data.toUInt(true); number = data.toUInt32BE(0);
return true; return true;
} }

View File

@@ -36,33 +36,30 @@
namespace Strawberry_TagLib { namespace Strawberry_TagLib {
namespace TagLib { namespace TagLib {
namespace Mod {
namespace Mod { class TAGLIB_EXPORT FileBase : public Strawberry_TagLib::TagLib::File {
protected:
explicit FileBase(FileName file);
explicit FileBase(IOStream *stream);
class TAGLIB_EXPORT FileBase : public Strawberry_TagLib::TagLib::File void writeString(const String &s, unsigned int size, char padding = 0);
{ void writeByte(unsigned char byte);
protected: void writeU16L(unsigned short number);
FileBase(FileName file); void writeU32L(unsigned int number);
FileBase(IOStream *stream); void writeU16B(unsigned short number);
void writeU32B(unsigned int number);
void writeString(const String &s, unsigned long size, char padding = 0); bool readString(String &s, unsigned int size);
void writeByte(unsigned char byte); bool readByte(unsigned char &_byte);
void writeU16L(unsigned short number); bool readU16L(unsigned short &number);
void writeU32L(unsigned long number); bool readU32L(unsigned int &number);
void writeU16B(unsigned short number); bool readU16B(unsigned short &number);
void writeU32B(unsigned long number); bool readU32B(unsigned int &number);
};
bool readString(String &s, unsigned long size); } // namespace Mod
bool readByte(unsigned char &byte); } // namespace TagLib
bool readU16L(unsigned short &number); } // namespace Strawberry_TagLib
bool readU32L(unsigned long &number);
bool readU16B(unsigned short &number);
bool readU32B(unsigned long &number);
};
}
}
}
#endif #endif

View File

@@ -24,44 +24,43 @@
// some helper-macros only used internally by (s3m|it|xm)file.cpp // some helper-macros only used internally by (s3m|it|xm)file.cpp
#define READ_ASSERT(cond) \ #define READ_ASSERT(cond) \
if(!(cond)) \ if (!(cond)) { \
{ \ setValid(false); \
setValid(false); \ return; \
return; \
} }
#define READ(setter,type,read) \ #define READ(setter, type, read) \
{ \ { \
type number; \ type number; \
READ_ASSERT(read(number)); \ READ_ASSERT(read(number)); \
setter(number); \ setter(number); \
} }
#define READ_BYTE(setter) READ(setter,unsigned char,readByte) #define READ_BYTE(setter) READ(setter, unsigned char, readByte)
#define READ_U16L(setter) READ(setter,unsigned short,readU16L) #define READ_U16L(setter) READ(setter, unsigned short, readU16L)
#define READ_U32L(setter) READ(setter,unsigned long,readU32L) #define READ_U32L(setter) READ(setter, unsigned int, readU32L)
#define READ_U16B(setter) READ(setter,unsigned short,readU16B) #define READ_U16B(setter) READ(setter, unsigned short, readU16B)
#define READ_U32B(setter) READ(setter,unsigned long,readU32B) #define READ_U32B(setter) READ(setter, unsigned int, readU32B)
#define READ_STRING(setter,size) \ #define READ_STRING(setter, size) \
{ \ { \
String s; \ String s; \
READ_ASSERT(readString(s, size)); \ READ_ASSERT(readString(s, size)); \
setter(s); \ setter(s); \
} }
#define READ_AS(type,name,read) \ #define READ_AS(type, name, read) \
type name = 0; \ type name = 0; \
READ_ASSERT(read(name)); READ_ASSERT(read(name));
#define READ_BYTE_AS(name) READ_AS(unsigned char,name,readByte) #define READ_BYTE_AS(name) READ_AS(unsigned char, name, readByte)
#define READ_U16L_AS(name) READ_AS(unsigned short,name,readU16L) #define READ_U16L_AS(name) READ_AS(unsigned short, name, readU16L)
#define READ_U32L_AS(name) READ_AS(unsigned long,name,readU32L) #define READ_U32L_AS(name) READ_AS(unsigned int, name, readU32L)
#define READ_U16B_AS(name) READ_AS(unsigned short,name,readU16B) #define READ_U16B_AS(name) READ_AS(unsigned short, name, readU16B)
#define READ_U32B_AS(name) READ_AS(unsigned long,name,readU32B) #define READ_U32B_AS(name) READ_AS(unsigned int, name, readU32B)
#define READ_STRING_AS(name,size) \ #define READ_STRING_AS(name, size) \
String name; \ String name; \
READ_ASSERT(readString(name, size)); READ_ASSERT(readString(name, size));
#endif #endif

Some files were not shown because too many files have changed in this diff Show More