mirror of
				https://github.com/arcan1s/awesome-widgets.git
				synced 2025-10-26 11:23:41 +00:00 
			
		
		
		
	Compare commits
	
		
			328 Commits
		
	
	
		
			V.3.0.0
			...
			88f70c0ea6
		
	
	| Author | SHA1 | Date | |
|---|---|---|---|
| 88f70c0ea6 | |||
| 338828da88 | |||
| 09a3c32d0c | |||
| e9beea2d7a | |||
| 1e69aa93f5 | |||
| 73f1617cbf | |||
| dea9d488df | |||
| 84de9755b5 | |||
| ff40ac96d8 | |||
| de9d90df4a | |||
| 883659272d | |||
| 0c443e9278 | |||
| 6ec30b71d1 | |||
| 4ed19a38d9 | |||
| e592de9555 | |||
| 1e0316c4de | |||
| 91abbdcd96 | |||
| e81d765098 | |||
| 8eaf444a25 | |||
| 183e7d8a20 | |||
| a4a1ae064f | |||
| 3e81cd16bb | |||
| 55319521b2 | |||
| e525cb4742 | |||
| 0f19bce80d | |||
| e0df575aa9 | |||
| 276248a748 | |||
| c05a87a540 | |||
| 9f46a17b98 | |||
| c2a18f8845 | |||
| 308b721c87 | |||
| c4c37406a5 | |||
| ffb66586d3 | |||
| 935fa04757 | |||
| fee70b7385 | |||
| ed21e3fb18 | |||
| 5807d4d75e | |||
| 5bd95fe41a | |||
| 13edbc5eb1 | |||
| 3b41239672 | |||
| b30bd35add | |||
| 4bba061d66 | |||
| ce30d90a23 | |||
| 423eabd857 | |||
| e5a9e99438 | |||
| eb911551eb | |||
| 5886dcc50b | |||
| 2d7d4c55a2 | |||
| ea5d73d8fd | |||
| 081f7a3c88 | |||
| 707fa727c5 | |||
| 5d0c05e856 | |||
| 4625afa0b9 | |||
| 5c56f813d2 | |||
| 5e08434835 | |||
| d2e6f2fe38 | |||
| e90e1da096 | |||
| 9d0a1bd70c | |||
| 7ce3e8b1cf | |||
| 7a00dce7c5 | |||
| 1a7530a847 | |||
| 1702045550 | |||
| fe88c72498 | |||
| 737759d03f | |||
| 934e30819c | |||
| a4d3a09f26 | |||
| acc4df92bc | |||
| 0a723aef82 | |||
| 271b8d55cf | |||
| 1afc33abef | |||
| 6b8244f3c4 | |||
| ff2032cca1 | |||
| 05aafc43e7 | |||
| d0c96ce829 | |||
| 6e62ceaac7 | |||
| 6e99e262d4 | |||
| 1b4d1d6944 | |||
| 8be4cc6e82 | |||
| 99c750fd0b | |||
| fca8019972 | |||
| 8d3dd0ed96 | |||
| 4974948151 | |||
| 9e50ade6a5 | |||
| 801eea9a8b | |||
| 1841fb5ed2 | |||
| 9c8dbdcb41 | |||
| 53afc07457 | |||
| 1caecb427b | |||
| ea10d526d5 | |||
| b818586a23 | |||
| a30e6ba21b | |||
| fe38d9133b | |||
| 38c5589c10 | |||
| ce09ab753b | |||
| 10f556e41b | |||
| d097e154ff | |||
| 8979ddf52b | |||
| d3dd9b3390 | |||
| 5615fd3ff9 | |||
| cbb8415a95 | |||
| 8dcdae9611 | |||
| bfdaadfc8b | |||
| 9543122816 | |||
| 6a24fc3f63 | |||
| 73d1b0a69f | |||
| 06f4882a2d | |||
| 7565ea2e82 | |||
| 004a97800c | |||
| 8dfb6b609f | |||
| f893b34e19 | |||
| a189e1af38 | |||
| 0332c59daf | |||
| 26b375254a | |||
| 89cb4b74fd | |||
| ea1d5fca86 | |||
| c3c46ff1b1 | |||
| 87406f70c4 | |||
| d520f55afc | |||
| 77a94aa0c1 | |||
| c103986f37 | |||
| 7568ae2a3b | |||
| 2ea5824fd7 | |||
| 47439ef151 | |||
| 63d19f038c | |||
| 9d5864aa62 | |||
| d74cd645fd | |||
| 80767c9dbf | |||
| 57ad8a0a2b | |||
| eea636c6f1 | |||
| 65141ff528 | |||
| f882756fad | |||
| 335fbb137e | |||
| fcb698e7fc | |||
| f1b5208bb2 | |||
| c9f5be6a60 | |||
| f7094ca187 | |||
| a326ebdabf | |||
| 3ee1d4c30e | |||
| 31c129163b | |||
| 80c994bce0 | |||
| 8aa6d0c626 | |||
| 33623fe75e | |||
| ecf4be8d9c | |||
| 7f6a2d8100 | |||
| 086870e9c2 | |||
| 1c52cf72be | |||
| 8e598bbd13 | |||
| 877f58784b | |||
| ee6cf60aa4 | |||
| 6449465be2 | |||
| 3403d1de50 | |||
| f9ddf690c7 | |||
| 87652eb774 | |||
| a9924a1432 | |||
| c5cb5cb359 | |||
| 65c8f552de | |||
| 486267256c | |||
| 94e87ee570 | |||
| b17aeb88e3 | |||
| 7d8036b0e1 | |||
| 4d1c05d737 | |||
| 95ede170d9 | |||
| 80d926290c | |||
| 5b9984d950 | |||
| e5b1102abf | |||
| f3f9239984 | |||
| 18c993c0d5 | |||
| a9e3e3f087 | |||
| 8f6d9e7ab5 | |||
| 22dcebe647 | |||
| 6e087f418f | |||
| b76cbb8ce3 | |||
| 24ad32b6ca | |||
| 51b7d3aa09 | |||
| 3674fba669 | |||
| f1116d1f83 | |||
| bcf757477b | |||
| 1864dc71d5 | |||
| 90c33ab24d | |||
| 052d8a034c | |||
| 064cd7e44f | |||
| 9e59b22c32 | |||
| 89f6661fa5 | |||
| 7836fbf77f | |||
| d90d9e2c6f | |||
| 927e93b7fc | |||
| fbf89f61b4 | |||
| 4adf457ada | |||
| 0d4211b2c4 | |||
| 085eec7a3d | |||
| baf5085506 | |||
| 17a0c61b4a | |||
| 3497922928 | |||
| 2f88c7ae60 | |||
| 8e8ac2f3c7 | |||
| 5cd5272f10 | |||
| bee8e2f180 | |||
| bf16e72e1e | |||
| faf259e339 | |||
| f6a6704fd2 | |||
| 4a0da3f978 | |||
| 71ae832bcd | |||
| 0f1763cb81 | |||
| b20a96d32f | |||
| 75d101cc8b | |||
| 5d9d551afe | |||
| 78b7a87c29 | |||
| bc2071a493 | |||
| 5a0541d06d | |||
| 18789f78b3 | |||
| ab01c9fa08 | |||
| 620c4bd1e3 | |||
| 4337379177 | |||
| 69c09d9ff8 | |||
| cae9e0d2e3 | |||
| 7b154af1b3 | |||
| fd3ed61191 | |||
| 2a257de1e6 | |||
| 6f09737f0f | |||
| b73fb19409 | |||
| c57a92b687 | |||
| 72f0d4587b | |||
| ea7c15d865 | |||
| 2dccc92933 | |||
| e71da9bdfb | |||
| f717c984b7 | |||
| 906ad56c46 | |||
| 301a908aed | |||
| 6bd7788aa9 | |||
| 51c7299ad0 | |||
| 326c65528d | |||
| 14aab3b758 | |||
| 7c37134aab | |||
| 734cbe2f4c | |||
| 4bcdb3a7ff | |||
| 5ad0c533f0 | |||
| d57e54b714 | |||
| 13f2d560d0 | |||
| 2220ad6bfe | |||
| 97f2e78308 | |||
| 3771ce6ec2 | |||
| 46db3fee4c | |||
| a339f9a743 | |||
| 4e52d16f1a | |||
| 097b63415e | |||
| 77388deab6 | |||
| d01d8f95e5 | |||
| 22586811de | |||
| 82dda14260 | |||
| f8e9945bc4 | |||
| 139554578c | |||
| 1827f641f9 | |||
| aef1a736c8 | |||
| 27a80274ac | |||
| 0e3f83f361 | |||
| d30df9dafc | |||
| f7a6b86cae | |||
| a3e7a84b92 | |||
| 354fd3cd0c | |||
| 15abe54290 | |||
| 60cbc40e9f | |||
| bfa4f9e98e | |||
| e9146b3b6c | |||
| d216ee1f79 | |||
| 81ceaa8160 | |||
| d4c7095d61 | |||
| d9409c25f4 | |||
| 7d1e035240 | |||
| fa795121aa | |||
| b9fda3e1cd | |||
| 1187c43e57 | |||
| 4a6aaa95b0 | |||
| 7e42c8cb49 | |||
| 88f0ebfe96 | |||
| 86458b8238 | |||
| bd65e44002 | |||
| fe7f82373b | |||
| 3a6033e676 | |||
| 453d4d5149 | |||
| 07c753b703 | |||
| 52b1255d3f | |||
| 7f665fef77 | |||
| 7ff1515c94 | |||
| 162708295d | |||
| 4a59acae09 | |||
| be9203e816 | |||
| 5af4b0c40c | |||
| 071d7fdb78 | |||
| 89d573450a | |||
| 50f3ef5bba | |||
| 8cae273ffb | |||
| 15d4d7667d | |||
| 95a5eec108 | |||
| 362f1fd87e | |||
| 5c474e822b | |||
| d856fa8e97 | |||
| 54e1545bb1 | |||
| 384097625c | |||
| 5ddf720a57 | |||
| 1610c190b4 | |||
| beb2682b04 | |||
| c1cf8185a3 | |||
| fb7fc4a104 | |||
| 210415cdff | |||
| bfac30c304 | |||
| 966c6059a0 | |||
| 1c78e0d779 | |||
| fba58c27e8 | |||
| 4b679ff570 | |||
| d8d2311621 | |||
| 809936a563 | |||
| 2c1aa1a4c9 | |||
| 02305b93d9 | |||
| aece0e2eff | |||
| c24e41d557 | |||
| 5be3f9246b | |||
| 81d99e66b9 | |||
| 940ef13ffa | |||
| 5ee1cec909 | |||
| 83ec82debd | |||
| ac089dee9c | |||
| 2d9ca281a9 | |||
| 99e38118a4 | |||
| 68dd2432cd | |||
| 919a10fff6 | |||
| d0ba320e1b | |||
| 8545192ef8 | |||
| f77cf7c1aa | 
							
								
								
									
										41
									
								
								.github/workflows/build.yml
									
									
									
									
										vendored
									
									
										Normal file
									
								
							
							
						
						
									
										41
									
								
								.github/workflows/build.yml
									
									
									
									
										vendored
									
									
										Normal file
									
								
							| @ -0,0 +1,41 @@ | |||||||
|  | name: build & tests | ||||||
|  |  | ||||||
|  | on: | ||||||
|  |   push: | ||||||
|  |     branches: [ development, master ] | ||||||
|  |   pull_request: | ||||||
|  |     branches: [ development, master ] | ||||||
|  |  | ||||||
|  | env: | ||||||
|  |   BUILD_TYPE: Release | ||||||
|  |  | ||||||
|  | jobs: | ||||||
|  |   run-tests: | ||||||
|  |      | ||||||
|  |     runs-on: ubuntu-latest | ||||||
|  |      | ||||||
|  |     container: | ||||||
|  |       image: archlinux:latest | ||||||
|  |       volumes: | ||||||
|  |         - ${{ github.workspace }}:/repo | ||||||
|  |       options: -w /repo | ||||||
|  |  | ||||||
|  |     steps: | ||||||
|  |     - uses: actions/checkout@v3 | ||||||
|  |  | ||||||
|  |     - name: create build environment | ||||||
|  |       run: pacman -Sy --noconfirm base-devel cmake extra-cmake-modules python util-linux-libs xorg-server-xvfb | ||||||
|  |      | ||||||
|  |     - name: install dependencies | ||||||
|  |       run: pacman -S --noconfirm plasma-workspace ksysguard | ||||||
|  |      | ||||||
|  |     - name: configure cmake | ||||||
|  |       run: cmake -B build -DKDE_INSTALL_USE_QT_SYS_PATHS=ON -DCMAKE_INSTALL_PREFIX=/usr -DCMAKE_BUILD_TYPE=${{env.BUILD_TYPE}} -DBUILD_FUTURE=ON -DBUILD_TESTING=ON sources | ||||||
|  |      | ||||||
|  |     - name: build | ||||||
|  |       working-directory: /repo/build | ||||||
|  |       run: make | ||||||
|  |      | ||||||
|  |     - name: test | ||||||
|  |       working-directory: /repo/build | ||||||
|  |       run: xvfb-run -a make test | ||||||
							
								
								
									
										51
									
								
								.github/workflows/release.yml
									
									
									
									
										vendored
									
									
										Normal file
									
								
							
							
						
						
									
										51
									
								
								.github/workflows/release.yml
									
									
									
									
										vendored
									
									
										Normal file
									
								
							| @ -0,0 +1,51 @@ | |||||||
|  | name: release | ||||||
|  |  | ||||||
|  | on: | ||||||
|  |   push: | ||||||
|  |     tags: | ||||||
|  |       - '*' | ||||||
|  |  | ||||||
|  | jobs: | ||||||
|  |   make-release: | ||||||
|  |  | ||||||
|  |     runs-on: ubuntu-latest | ||||||
|  |  | ||||||
|  |     steps: | ||||||
|  |       - uses: actions/checkout@v3 | ||||||
|  |  | ||||||
|  |       - name: create changelog | ||||||
|  |         id: changelog | ||||||
|  |         uses: jaywcjlove/changelog-generator@main | ||||||
|  |         with: | ||||||
|  |           token: ${{ secrets.GITHUB_TOKEN }} | ||||||
|  |           filter: 'Release \d+\.\d+\.\d+' | ||||||
|  |  | ||||||
|  |       - name: create archive | ||||||
|  |         run: bash create_archive.sh | ||||||
|  |         env: | ||||||
|  |           VERSION: ${{ steps.version.outputs.VERSION }} | ||||||
|  |  | ||||||
|  |       - name: build debian package | ||||||
|  |         run: | | ||||||
|  |           sudo apt update && \ | ||||||
|  |           sudo apt install -yq cmake extra-cmake-modules g++ git gettext make && \ | ||||||
|  |           sudo apt install -yq libkf5i18n-dev libkf5notifications-dev libkf5service-dev \ | ||||||
|  |             libkf5windowsystem-dev libkf5plasma-dev qtbase5-dev qtdeclarative5-dev \ | ||||||
|  |             plasma-workspace-dev && \ | ||||||
|  |           cmake -B build-deb -DKDE_INSTALL_USE_QT_SYS_PATHS=ON -DCMAKE_INSTALL_PREFIX=/usr -DCMAKE_BUILD_TYPE=Optimization -DBUILD_FUTURE=ON -DBUILD_DEB_PACKAGE=ON sources && \ | ||||||
|  |           cd build-deb && \ | ||||||
|  |           make package && \ | ||||||
|  |           cd .. | ||||||
|  |  | ||||||
|  |       - name: release | ||||||
|  |         uses: softprops/action-gh-release@v1 | ||||||
|  |         with: | ||||||
|  |           body: | | ||||||
|  |             ${{ steps.changelog.outputs.compareurl }} | ||||||
|  |             ${{ steps.changelog.outputs.changelog }} | ||||||
|  |           files: | | ||||||
|  |             awesome-widgets-*-src.tar.xz | ||||||
|  |             build-deb/plasma-widget-awesome-widgets-*.deb | ||||||
|  |           fail_on_unmatched_files: true | ||||||
|  |         env: | ||||||
|  |           GITHUB_TOKEN: ${{ secrets.GITHUB_TOKEN }} | ||||||
							
								
								
									
										2
									
								
								.gitignore
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										2
									
								
								.gitignore
									
									
									
									
										vendored
									
									
								
							| @ -43,6 +43,8 @@ build | |||||||
| *pkg.tar.[gx]z | *pkg.tar.[gx]z | ||||||
| src | src | ||||||
| pkg | pkg | ||||||
|  | *.deb | ||||||
|  |  | ||||||
| # clion settings | # clion settings | ||||||
| .idea | .idea | ||||||
|  | cmake-build* | ||||||
|  | |||||||
							
								
								
									
										9
									
								
								.gitmodules
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										9
									
								
								.gitmodules
									
									
									
									
										vendored
									
									
								
							| @ -1,9 +0,0 @@ | |||||||
| [submodule "sources/3rdparty/task"] |  | ||||||
| 	path = sources/3rdparty/task |  | ||||||
| 	url = https://github.com/arcan1s/qtadds-taskadds-qprocess.git |  | ||||||
| [submodule "sources/3rdparty/tasks"] |  | ||||||
| 	path = sources/3rdparty/tasks |  | ||||||
| 	url = https://github.com/mhogomchungu/tasks.git |  | ||||||
| [submodule "sources/3rdparty/fontdialog"] |  | ||||||
| 	path = sources/3rdparty/fontdialog |  | ||||||
| 	url = https://github.com/arcan1s/qtadds-fontdialog.git |  | ||||||
| @ -1,11 +0,0 @@ | |||||||
| install: |  | ||||||
|   - sudo apt-get update -qq |  | ||||||
|   - sudo apt-get install -y cmake extra-cmake-modules g++ git libkf5i18n-dev libkf5notifications-dev libkf5service-dev libkf5windowsystem-dev plasma-framework-dev qtbase5-dev qtdeclarative5-dev |  | ||||||
|   - git submobule update --init |  | ||||||
|   - rm -rf build |  | ||||||
|   - mkdir build |  | ||||||
|  |  | ||||||
| script: |  | ||||||
|   - cd build |  | ||||||
|   - cmake -DCMAKE_INSTALL_PREFIX=/usr -DCMAKE_BUILD_TYPE=Release -DKDE_INSTALL_USE_QT_SYS_PATHS=ON ../sources |  | ||||||
|   - make |  | ||||||
							
								
								
									
										8
									
								
								.tx/config
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										8
									
								
								.tx/config
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,8 @@ | |||||||
|  | [main] | ||||||
|  | host = https://www.transifex.com | ||||||
|  |  | ||||||
|  | [awesome-widgets.awesome-widgetspot] | ||||||
|  | source_file = sources/translations/awesome-widgets.pot | ||||||
|  | source_lang = en | ||||||
|  | type = PO | ||||||
|  | file_filter = sources/translations/<lang>.po | ||||||
							
								
								
									
										16
									
								
								AUTHORS
									
									
									
									
									
								
							
							
						
						
									
										16
									
								
								AUTHORS
									
									
									
									
									
								
							| @ -1,9 +1,15 @@ | |||||||
| Current developers: | Current developers: | ||||||
| Evgeniy Alekseev aka arcanis <esalexeev (at) gmail (dot) com> | Evgeniy Alekseev aka arcanis <esalexeev (at) gmail (dot) com> | ||||||
|  |  | ||||||
|  | Packagers: | ||||||
|  | Konstantin Voinov (openSuSe) | ||||||
|  |  | ||||||
| Translators: | Translators: | ||||||
| @Mermouy (French translation) | @Mermouy (French) | ||||||
| Ernesto Avilés Vzqz (Spanish translation) | Ernesto Avilés Vzqz (Spanish) | ||||||
| @underr (Brazillian Portuguese translation) | @underr (Brazillian Portuguese) | ||||||
| Виктор Слободян (Ukrainian translation) | Виктор Слободян (Ukrainian) | ||||||
| Steve Lemuel (Chinese translation) | Steve Lemuel (Chinese) | ||||||
|  | Mariusz Kocoń (Polish) | ||||||
|  | Ibnu Daru Aji (Indonesian) | ||||||
|  | Antonio Vivace (Italian) | ||||||
|  | |||||||
							
								
								
									
										120
									
								
								CHANGELOG
									
									
									
									
									
								
							
							
						
						
									
										120
									
								
								CHANGELOG
									
									
									
									
									
								
							| @ -1,3 +1,123 @@ | |||||||
|  | Ver.3.5.0: | ||||||
|  | + wayland support | ||||||
|  | * update code to latest standards | ||||||
|  | - drop support of windows preview | ||||||
|  |  | ||||||
|  | Ver.3.4.2: | ||||||
|  | + Italian translation (#136, thanks to @avivace) | ||||||
|  | + stooq quotes support (default) (#131) | ||||||
|  | + Indonesian translation (#132, thanks to @ibnuda) | ||||||
|  | + new keys - $batleftN, $batnowN, $batrateN, $battotalN, $brightness, $volume | ||||||
|  | * fix compiler warnings | ||||||
|  | * update to new qt api | ||||||
|  | - fix non printable spaces (#142, #143) | ||||||
|  |  | ||||||
|  |  | ||||||
|  | Ver.3.3.3: | ||||||
|  | + add custom keys support (#101) | ||||||
|  | * DBus interface improvements | ||||||
|  | * rename tags (up|down|ps)total.* to (up|down|ps)tot.* | ||||||
|  | - fix issue with invalid (up|down)total($|kb) calculation (#127) | ||||||
|  | - fix issue with wrong applet identation (#125) | ||||||
|  |  | ||||||
|  | Ver.3.3.2: | ||||||
|  | - fix bug with invalid DP colour configuration | ||||||
|  | - fix bug with invalid HTML tags operation | ||||||
|  | * update to newest Qt and plasma | ||||||
|  |  | ||||||
|  | Ver.3.3.1: | ||||||
|  | + add ability to direct issue report (#104) | ||||||
|  | + add key completion (#105) | ||||||
|  | + add ability to load historical configuration (#106) | ||||||
|  | + add ability to upload configuration to remote server (#106) | ||||||
|  | + add stylish (#112) | ||||||
|  | + add DBus intergration (#115) | ||||||
|  | + add cron-like extenstion triggers (`X-AW-Socket` option) (#117) | ||||||
|  | + add ability to trigger extensions by calling socket (`X-AW-Schedule` option) (#118) | ||||||
|  | + add source for network requests | ||||||
|  | + add Json formatter | ||||||
|  | - drop patches provided support for Qt 5.4 | ||||||
|  | * use custom signal slot connection for dataengine (`BUILD_FUTURE` flag required) | ||||||
|  | * update UI | ||||||
|  | * refactoring | ||||||
|  |  | ||||||
|  | Ver.3.2.1: | ||||||
|  | + add X-AW-Translate option to DateTime formatter | ||||||
|  | + add formatters to configuration export and import | ||||||
|  | + add macro fuction to selector | ||||||
|  | + add default formatters | ||||||
|  | - add translation reinit (fix #102) | ||||||
|  | - fix invalid float formatter saving | ||||||
|  | - fix missing ApiVersion for formatters | ||||||
|  | * use sockets for MPD data | ||||||
|  | * translation update | ||||||
|  |  | ||||||
|  | Ver.3.2.0: | ||||||
|  | + custom formatters (#91) | ||||||
|  | + add backend tests (#95) | ||||||
|  | + add OWM weather provider for ExtWeather. It can be switched by using X-AW-Provider (ApiVer 3) | ||||||
|  | + add DBus timeout for calls (#96) | ||||||
|  | + add macros support (`$aw_macro` and `$aw_macro_*` for calls) | ||||||
|  | - fix vertical alignment (#94) | ||||||
|  | - fix bug with invalid updates on `X-AW-Interval=1` for ExtScript and ExtUpgrade | ||||||
|  | - fix bug with missing derivate vaules in ExtQuotes | ||||||
|  | - fix possible plasma crash (#96) | ||||||
|  | - fix possible undefined behaviour if no extensions was found by name | ||||||
|  | * split dataengine sources to own library | ||||||
|  | * allow to use screened double brakets inside functions | ||||||
|  |  | ||||||
|  | Ver.3.1.2: | ||||||
|  | + add standalone widget configuration example | ||||||
|  | + new tag tstime | ||||||
|  | + new tags uptotal* and downtotal* | ||||||
|  | + add background configuration button | ||||||
|  | + allow X-AW-Direction property works with graphs and bars (graphical items) | ||||||
|  | + add bars support to Graphical Items | ||||||
|  | + add integration with transifex | ||||||
|  | + add integration with coverity | ||||||
|  | - drop update source (breaking changes if you are using it as developer) | ||||||
|  | - fix pen generation from file image (graphical items) | ||||||
|  | - fix bug with which font html parameters will be pasted even if there cancel clicked | ||||||
|  | - fix bug with invalid total memory formating | ||||||
|  | - fix bug in configuration UI with which there is no selected text pasted | ||||||
|  | - fixes in graphical item configuration UI and behaviour | ||||||
|  | - fix changelog building | ||||||
|  | - fix invalid cast (found by coverity) | ||||||
|  | * change text update mechanism | ||||||
|  | * grahical item improvements | ||||||
|  | * code improvements | ||||||
|  | * translation update | ||||||
|  |  | ||||||
|  | Ver.3.1.1: | ||||||
|  | - fix broken gcc build (#87) | ||||||
|  | * translation update | ||||||
|  |  | ||||||
|  | Ver.3.1.0: | ||||||
|  | + implement templates support (#71) | ||||||
|  | + implement special functions (#71) | ||||||
|  | + special directory for configs (#72) | ||||||
|  | + custom bar images (#80) | ||||||
|  | + custom bar values (#80) | ||||||
|  | + show changelog after updates | ||||||
|  | + add Polish translation (thanks to Mariusz Kocoń) | ||||||
|  | + use Qt-5.6 features | ||||||
|  | - fix invalid tooltip resize in desktop panel (#74) | ||||||
|  | - fix graphical items blinking (#81) | ||||||
|  | - drop tasks and move to native QProcess to avoid crash (#83) | ||||||
|  | - the newest Qt (5.6) fixes | ||||||
|  | - the newest Plasma fixes | ||||||
|  | * move from OWM to Yahoo! Weather (#73) | ||||||
|  | * improve performance by using optimized subscriptions (#75) | ||||||
|  | * improve performance by storing QVariant instead of frequent casting (#75) | ||||||
|  | * change bar names semantic to the simplest one (#80, breaking changes) | ||||||
|  | * high memory usage notifications have been changed from 90 to 80 perc | ||||||
|  | * a large part of refactoring | ||||||
|  |  | ||||||
|  | Ver.3.0.1: | ||||||
|  | + add patches for old Qt versions | ||||||
|  | - drop `nullptr` checking | ||||||
|  | * move dialogs to QML part | ||||||
|  |  | ||||||
| Ver.3.0.0: | Ver.3.0.0: | ||||||
| + add tags upunits, downunits, upkb, downkb | + add tags upunits, downunits, upkb, downkb | ||||||
| + add tags dalbum, dartist, dtitle, salbum, sartist, stitle | + add tags dalbum, dartist, dtitle, salbum, sartist, stitle | ||||||
|  | |||||||
							
								
								
									
										378
									
								
								CHANGELOG-RU
									
									
									
									
									
								
							
							
						
						
									
										378
									
								
								CHANGELOG-RU
									
									
									
									
									
								
							| @ -1,378 +0,0 @@ | |||||||
| Вер.3.0.0: |  | ||||||
| + добавлены теги upunits, downunits, upkb, downkb |  | ||||||
| + добавлены теги dalbum, dartist, dtitle, salbum, sartist, stitle |  | ||||||
| + добавлена поддержка лямбда функций |  | ||||||
| + добавлен голландский перевод (спасибо Heimen Stoffels) |  | ||||||
| + добавлена опциональная проверка обновлений на старте |  | ||||||
| + добавлена возможность отключить перевод строковых тегов |  | ||||||
| + добавлена фильтрация в ExtUpgrade (ApiVer == 3) |  | ||||||
| + добавлена возможность смены иконки погоды (ApiVer == 2) |  | ||||||
| + показ сообщений, если не найдено обновлений, при прямом запросе |  | ||||||
| + добавлено перенаправление swap в ExtScript (ApiVer == 4) |  | ||||||
| + добавлена возможность переноса слов |  | ||||||
| + добавлена поддержка графиков в GraphicalItems (ApiVer == 3) |  | ||||||
| + добавлена возможность показать превью из настроек |  | ||||||
| + добавлены импорт и экспорт конфигурации |  | ||||||
| - исправлен ворнинг для пустых тултипов |  | ||||||
| - исправлен баг, приводящий к увеличению виджета |  | ||||||
| - исправлен баг с отсутствием сохранения настроек DE |  | ||||||
| - исправлен баг с большим размером на старте |  | ||||||
| - исправлен баг с невозможностью использования более, чем 10 расширений |  | ||||||
| - убраны уведомления о свободном месте на диске |  | ||||||
| - убран ключ X-AW-Output в ExtScript (ApiVer == 4) |  | ||||||
| - исправлен возможное падение при вызове деструктора web-источников |  | ||||||
| * более удобный интерфейс настройки |  | ||||||
| * улучшено логирование, применен стиль логирования Qt5 |  | ||||||
| * переключено на конкурентное обновление (-DBUILD_FUTURE=ON) |  | ||||||
| * немного оптимизаций |  | ||||||
| * лучшее определение активного сетевого устройства |  | ||||||
| * значение котировок типа double (ApiVer == 3) |  | ||||||
| * лучший показ котировок |  | ||||||
| * больше комментариев в коде |  | ||||||
| * большее следование стилю кода Qt |  | ||||||
| * настройки интервала перемещены в другую группу |  | ||||||
| * работа с DataEngine была перемещена в плагин |  | ||||||
| * изменен формат данных внутри DataEngine (ломает совместимость) |  | ||||||
|  |  | ||||||
| Вер.2.4.0: |  | ||||||
| + добавлена поддержка погоды |  | ||||||
| + добавлена поддержка LA (теги la1, la5, la15) |  | ||||||
| + добавлена кнопка "Сбросить кеш" |  | ||||||
| + добавлена поддержка сборки с clang |  | ||||||
| + добавлен ExtItem агрегатор |  | ||||||
| + добавлено время в дебаг вывод |  | ||||||
| + добавлена возможность отключить монитор плеера |  | ||||||
| - исправлен баг с неправильным расчетам скорости сети |  | ||||||
| - исправлен баг с неправильной инициализацией диалога шрифтов |  | ||||||
| - исправлен баг с отсутствием удаления объектов (#60) |  | ||||||
| * переписаны ExtItems для использования собственного абстрактного класса |  | ||||||
| * улучшена работа с обновлениями виджета (#57) |  | ||||||
| * рисование интерфейса переписано на модель сигнал-слот |  | ||||||
| * плагин опять стал приватным |  | ||||||
| * исправлен каст плагина с синглтона (#57) |  | ||||||
| * изменена логика показа данных в DataEngine |  | ||||||
| * изменена сборка пакета и прекращена поддержка отдельного пакета с DataEngine |  | ||||||
| * замена QMap на QHash, если возможно |  | ||||||
| * замена циклов на итераторы, если возможно |  | ||||||
| * список рабочий столов сейчас представлен, как QStringList |  | ||||||
| * рефакторинг |  | ||||||
|  |  | ||||||
| Вер.2.3.3: |  | ||||||
| * изменен рендеринг с Qt на нативный |  | ||||||
|  |  | ||||||
| Вер.2.3.2: |  | ||||||
| + добавлено изменение скорости интернета с KB/s на MB/s, если величина более 1000 KB/s |  | ||||||
| - пофикшен баг с отсутствием действия при активации ссылки (спасибо infans) |  | ||||||
| * улучшена поддержка котировок, например пофикшен возможный сегфолт плазмы |  | ||||||
| * переписать основной виджет к модели сигнал-слот |  | ||||||
| * улучшена поддержка расчета заряда батареи для нескольких батарей |  | ||||||
|  |  | ||||||
| Вер.2.3.1: |  | ||||||
| - пофикшен баг с отсутствием обновлений значений из настроек |  | ||||||
| - убрана поддержка фильтра `\t` |  | ||||||
| * обновлены переводы |  | ||||||
| * теги fan были объединены с temp, улучшена поддержка тегов lmsensors |  | ||||||
|  |  | ||||||
| Вер.2.3.0: |  | ||||||
| + добавлена поддержка фильтров в ExtScript (ApiVer = 3) (#53) |  | ||||||
| + добавлена возможность отображать `\n`, как перенос строки (#53) |  | ||||||
| + добавлена сортировка ключей в комбобоксах |  | ||||||
| + добавлена поддержка тегов (perc)(bid|ask|price)chg (ApiVer = 2) |  | ||||||
| + добавлены таймауты запросов котировок (#54) |  | ||||||
| - исправлен баг с неправильным парсингом html |  | ||||||
| * ExtQuotes теперь использует JSON вместо XML (ApiVer = 2) |  | ||||||
| * более красивый UI настроек |  | ||||||
| * некоторый рефакторинг |  | ||||||
|  |  | ||||||
| Вер.2.2.2: |  | ||||||
| + добавлена поддержка котировок (#46) |  | ||||||
| + добавлена поддержка свойств height и width (#47) |  | ||||||
| + добавлена поддержка перевода в плагины (#50) |  | ||||||
| + добавлено кеширование ключей |  | ||||||
| + добавлена поддержка X-AW-Number в ExtScript, ExtUpgrade (ApiVer=2) |  | ||||||
| + добавлено переподключение DataEngine (в некоторых случаях отдельные значения не обновляются) |  | ||||||
| + добавлено обновление ширины и высоты виджета |  | ||||||
| - исправлен баг при обновлении Ext* (#49) |  | ||||||
| - исправлен баг с сохранением настроек GPUDEV, HDDDEV из графического интерфейса |  | ||||||
| - исправлены настройки Desktop Panel |  | ||||||
| - исправлен баг с добавлением настроек параграфа |  | ||||||
| * дропнута поддержка KDE4 версии (отдельная ветка разработки) |  | ||||||
| * увеличены лимиты для интервалов (#48) |  | ||||||
| * hddtemp был заменен на smartctl по умолчанию |  | ||||||
| * плагины были перемещены из private |  | ||||||
| * более корректное рисование интерфейса |  | ||||||
|  |  | ||||||
| Вер.2.2.1: |  | ||||||
| + добавлен ключ X-AW-Interval в ExtUpgrade (ApiVer = 1, default = 3600) |  | ||||||
| + добавлено кеширование значений в ExtScript и ExtUpgrade |  | ||||||
| + добавлена поддержка isActive в ExtUpgrade |  | ||||||
| * изменена проверка версий cmake |  | ||||||
| * улучшена логика работы ExtScript и ExtUpgrade |  | ||||||
|  |  | ||||||
| Вер.2.2.0: |  | ||||||
| * портирование на Plasma 5 |  | ||||||
| + добавлена поддержка списка окон в тултипе |  | ||||||
| + добавлены уведомления на системные события |  | ||||||
| + добавлена возможность показать выбранный тег |  | ||||||
| + добавлены теги $downN, $upN |  | ||||||
| + добавлен инструмент netdev |  | ||||||
| + добавлена поддержка smartctl |  | ||||||
| - убрана возможность сокрытия панелей |  | ||||||
| * незначительно изменена логика работы виджета |  | ||||||
| * исправлена отрисовка превью окон |  | ||||||
| * обновление кодовой базы KDE4 под новый ABI |  | ||||||
| * оптимизация работы DataEngine |  | ||||||
| * PKGCMD были заменены на ExtUpgrade |  | ||||||
|  |  | ||||||
| Вер.2.1.0: |  | ||||||
| + добавлена функция автообновления |  | ||||||
| + добавлена поддерка кастомных скриптов |  | ||||||
| + добавлена поддержка превью рабочего стола для DP |  | ||||||
| + добавлена поддержка баров |  | ||||||
| - удалена поддержка кастомных команд |  | ||||||
| - фикс #34 |  | ||||||
| * исправлены уведомления |  | ||||||
| * переписан UI в соответствии со стандартами KDE |  | ||||||
| * исправлены границы для графических объектов |  | ||||||
| * переписан DP и DE на использование библиотечных функций |  | ||||||
| * изменено определение свойств GPU |  | ||||||
| * изменен поиск регулярных выражений |  | ||||||
|  |  | ||||||
| Вер.2.0.5: |  | ||||||
| + добавлен нижний край к тултипу (#30) |  | ||||||
| * исправлен баг с обновлением сетевого интерфейса |  | ||||||
| * исправлен баг с тултипом батареи |  | ||||||
| * отредактирована разметка DP (#31) |  | ||||||
|  |  | ||||||
| Вер.2.0.4: |  | ||||||
| * более корректная работа с батареями |  | ||||||
|  |  | ||||||
| Вер.2.0.3: |  | ||||||
| + добавлена поддержка нескольких тегов $bat |  | ||||||
| * отредактирована разметка |  | ||||||
|  |  | ||||||
| Вер.2.0.2: |  | ||||||
| + добавлены теги $fan* |  | ||||||
| + добавлен китайский перевод (спасибо @Lemueler) |  | ||||||
| + добавлена проверка сетевых устройств на p2p |  | ||||||
| - удалено ps stats из уведомлений |  | ||||||
| * еще один фикс тегов cpu* и cpucl |  | ||||||
| * переписан конфигурационный интерфейс DP |  | ||||||
| * исправлены нестандартные единицы температуры |  | ||||||
|  |  | ||||||
| Вер.2.0.1: |  | ||||||
| + добавлена поддержка кастомных интерфейсов |  | ||||||
| * фикс затенения краев поля |  | ||||||
| * фикс тегов cpu*, cpucl*, gpu*, ps* |  | ||||||
|  |  | ||||||
| Вер.2.0.0: |  | ||||||
| + добавлен виджет awesome widget |  | ||||||
| + новые теги $hddfreemb, $hddfreegb, $memusedmb, $memusedgb, $memfreemb $memfreegb, $swapfreemb, $swapfreegb |  | ||||||
| + новый DataEngine - battery |  | ||||||
| + новый тултип - battery |  | ||||||
| + новые температурные единицы см^-1, ккал/моль, кДж/моль, Реамюр |  | ||||||
| + добавлена возможность скрывать панели по хоткею |  | ||||||
| - удален виджет pytextmonitor |  | ||||||
| - убраны отдельные поля |  | ||||||
| - удалены README |  | ||||||
| * проект переименован в Awesome Widgets |  | ||||||
| * тег $custom (время) переименован в $ctime |  | ||||||
| * тег $custom (время работы) переименован в $cuptime |  | ||||||
| * тег $time (плеер) переименован в $duration |  | ||||||
| * тег $number переименован в $ndesktop |  | ||||||
| * тег $name переименован в $desktop |  | ||||||
| * тег $total переименован в $tdesktops |  | ||||||
| * дебаг флаг теперь DEBUG |  | ||||||
| * проект полностью переписан на C++ |  | ||||||
| * тултип независит теперь от самого виджета |  | ||||||
| * настройка выравнивания текста теперь осуществляется только HTML тегами |  | ||||||
| * настройки батареи и адаптора питания вынесены в DataEngine |  | ||||||
| * изменения в конфигурационном интерфейсе |  | ||||||
|  |  | ||||||
| Вер.1.11.0: |  | ||||||
| + добавлена возможность отключения фона |  | ||||||
| + добавлен desktop DataEngine |  | ||||||
| + добавлен виджет PTM desktop panel |  | ||||||
| + добавлено поле текущий рабочий стол (теги $name, $number, $total) |  | ||||||
| + добавлена возможность взаимодействия с плеерами через mpris |  | ||||||
| * добавлено использование сабмодулей |  | ||||||
| * DataEngine переписан с использованием tasks |  | ||||||
| * фикс чтения температуры nvidia |  | ||||||
|  |  | ||||||
| Вер.1.10.2: |  | ||||||
| - фикс ошибки с tempUnits в поле hddtemp |  | ||||||
|  |  | ||||||
| Вер.1.10.1: |  | ||||||
| - пофикшена ошибка с форматами чисел в поле ps |  | ||||||
| - пофикшена инициализация значений в DataEngine |  | ||||||
| - пофикшена проверка qmmp |  | ||||||
|  |  | ||||||
| Вер.1.10.0: |  | ||||||
| + добавлены теги $hddtotmb, $hddtotgb |  | ||||||
| + добавлены теги $memtotmb, $memtotgb, $swaptotmb, $swaptotgb |  | ||||||
| + добавлена возможность форсированного обновления |  | ||||||
| + добавлена поддержка clementine |  | ||||||
| + добавлена поддержка команды hddtemp |  | ||||||
| + добавлены поддержка disable для HDDDEV и GPUDEV |  | ||||||
| + добавлена многопоточность в DataEngine |  | ||||||
| + добавлена проверка ошибок для переменных DataEngine |  | ||||||
| + добавлена возможность отключения всплывающих сообщений |  | ||||||
| + добавлена возможность использовать вертикальную разметку |  | ||||||
| + добавлена возможность добавить пустое пространство |  | ||||||
| + добавлена возможность вывода отладочной информации (установите переменную окружения 'PTM_DEBUG' в 'yes') |  | ||||||
| + добавлен перевод Brazillian Portuguese (@underr) |  | ||||||
| + добавлена украинская локализация (Виктор Слободян) |  | ||||||
| - удален util.py |  | ||||||
| - удалены функции createNotifyrc, поскольку они больше не нужны |  | ||||||
| * рефакторинг конфигурационного интерфейса |  | ||||||
| * настройки плеера были перенесены из виджета в DataEngine |  | ||||||
| * исправлен вызов readme под четырехсимвольное обозначение |  | ||||||
| * перенос readme с md на html |  | ||||||
|  |  | ||||||
| Вер.1.9.0: |  | ||||||
| + добавлена поддержка нескольких кастомных команд ($customN) |  | ||||||
| + добавлено поле скорости hdd ($hddrN, $hddwN) |  | ||||||
| + добавлены действия в контекстное меню (открыть readme и запустить ksysguard) |  | ||||||
| * исправлен парсинг поля pkg |  | ||||||
| - убрано событие по двойному клику |  | ||||||
|  |  | ||||||
| Вер.1.8.3: |  | ||||||
| + добавлен испанский перевод (Ernesto Avilés Vzqz) |  | ||||||
| + добавлены файлы readme |  | ||||||
|  |  | ||||||
| Вер.1.8.2: |  | ||||||
| + добавлены различные температурные единицы (Farenheit и Kelvin) |  | ||||||
|  |  | ||||||
| Вер.1.8.1: |  | ||||||
| * поле pkg теперь обновляется раз в час |  | ||||||
|  |  | ||||||
| Вер.1.8.0: |  | ||||||
| + добавлены скроллбары в конфигурационный интерфейс |  | ||||||
| + добавлена настройка значений AC |  | ||||||
| + добавлены теги $hddmbN, $hddgbN |  | ||||||
| + добавлена поддержка ps (теги $ps, $pscount, $pstotal) |  | ||||||
| + добавлена поддержка пакетных менеджеров (теги $pkgcountN) |  | ||||||
| * рефакторинг DataEngine |  | ||||||
|  |  | ||||||
| Вер.1.7.4: |  | ||||||
| + добавлена поддержка более чем 10 тэгов в полях cpu, cpuclock, hdd, hddtemp, temp |  | ||||||
| - фиксы в форматах |  | ||||||
| * изменения в кастомном формате аптайма |  | ||||||
|  |  | ||||||
| Вер.1.7.3: |  | ||||||
| + добавлены тэги $memgb, $swapgb |  | ||||||
| + добавлена директория readme |  | ||||||
| + добавлен французский перевод (@Mermouy) |  | ||||||
| - фикс подсказки для кастомного формата аптайма |  | ||||||
| - фикс #12 |  | ||||||
| * больше рефакторинга богу рефакторинга |  | ||||||
| * интервал обновления сетевого интерфейса теперь 30*interval |  | ||||||
| * тэги $swap, $swapmb и $swapgb теперь работают одновременно (тэги $mem* тоже) |  | ||||||
| * player_name теперь имеет тип str |  | ||||||
|  |  | ||||||
| Вер.1.7.2: |  | ||||||
| + добавлены тэги $dd, $d, $hh, $h, $mm, $m в кастомный формат аптайма |  | ||||||
| + добавлено масштабирование в тултипы |  | ||||||
| + добавлена конфигурация DE из виджета |  | ||||||
| + добавлена кастомная команда в DE |  | ||||||
| - удалена кастомная команда из виджета |  | ||||||
| - удалены тэги $ds, $hs, $ms |  | ||||||
|  |  | ||||||
| Вер.1.7.1: |  | ||||||
| - удалено поле с ошибкой |  | ||||||
| * пофикшен баг с нулевым значением в тултипе |  | ||||||
|  |  | ||||||
| Вер.1.7.0: |  | ||||||
| + добавлена поддержка нескольких тэгов в поле температуры |  | ||||||
| + добавлены тэги $cpuN, $cpuclN, $hddN, $tempN, $hddtempN, $up, $down |  | ||||||
| + добавлено поле с кастомной командой |  | ||||||
| + добавлены тултипы для полей CPU, частоты CPU, памяти, swap и сеть |  | ||||||
| - удалены тэги @@...@@, $temp, $ccpu, $ccpucl, $net |  | ||||||
| * изменения в настройках |  | ||||||
| * виджет не требует больше net-tools |  | ||||||
| * рефакторинг |  | ||||||
|  |  | ||||||
| Вер.1.6.1: |  | ||||||
| + добавлена поддержка нескольких девайсов в поле hddtemp |  | ||||||
| - фикс #6 |  | ||||||
|  |  | ||||||
| Вер.1.6.0: |  | ||||||
| + добавлен тэг $custom в поля time и uptime (#8) |  | ||||||
| + русский перевод |  | ||||||
| - фикс #6, #10 |  | ||||||
| * лицензия изменена на GPLv3 |  | ||||||
| * изменения в сборке проекта |  | ||||||
|  |  | ||||||
| Вер.1.5.3: |  | ||||||
| + добавлены тэги $album, $progress и $time tags в поле плеера (#7) |  | ||||||
| - фикс бага с нулевым свопом |  | ||||||
|  |  | ||||||
| Вер.1.5.2: |  | ||||||
| + добавлены тэги $isotime, $shorttime и $longtime в поле time |  | ||||||
|  |  | ||||||
| Вер.1.5.1: |  | ||||||
| + добавлено поле time |  | ||||||
|  |  | ||||||
| Вер.1.5.0: |  | ||||||
| + добавлено поле player |  | ||||||
|  |  | ||||||
| Вер.1.4.0: |  | ||||||
| + добавлены уведомления |  | ||||||
|  |  | ||||||
| Вер.1.3.6: |  | ||||||
| - фикс #1, #2 |  | ||||||
|  |  | ||||||
| Вер.1.3.5: |  | ||||||
| - фикс бага с определением сетевого устройства |  | ||||||
|  |  | ||||||
| Вер.1.3.4: |  | ||||||
| + добавлено событие по двойному щелчку |  | ||||||
| + добавлен автовыбор сетевого устройства |  | ||||||
| - фикс еще одного падения Плазмы |  | ||||||
|  |  | ||||||
| Вер.1.3.3: |  | ||||||
| * редактирование чтения температурного устройства |  | ||||||
|  |  | ||||||
| Вер.1.3.2b: |  | ||||||
| - фикс падения Плазмы |  | ||||||
|  |  | ||||||
| Вер.1.3.2: |  | ||||||
| + добавлены поля GPU, GPU temp, HDD и HDD temp |  | ||||||
| + добавлены тэги $ccpu и $ccpucl |  | ||||||
| - фикс некоторых багов |  | ||||||
| * оптимизация и рефакторинг |  | ||||||
|  |  | ||||||
| Вер.1.2.2: |  | ||||||
| + добавлен тултип в строки |  | ||||||
|  |  | ||||||
| Вер.1.2.1: |  | ||||||
| * если батарея не существует, возвращает off |  | ||||||
| * если AC не существует, возвращает '(?)' |  | ||||||
|  |  | ||||||
| Вер.1.2.0: |  | ||||||
| + добавлен тэг $ac в поле battery |  | ||||||
| + добавлена настройка устройств батареи и AC |  | ||||||
| + добавлена проверка ошибок |  | ||||||
| * тэг $bat теперь читается из /sys/* |  | ||||||
|  |  | ||||||
| Вер.1.1.4: |  | ||||||
| * изменения в парсинге acpi (поле battery) |  | ||||||
|  |  | ||||||
| Вер.1.1.3: |  | ||||||
| + добавлена настройка ширины шрифта |  | ||||||
| - небольшие фиксы |  | ||||||
| * отредактирована настройка шрифта. Теперь настройки для шрифта работают нормально |  | ||||||
|  |  | ||||||
| Вер.1.1.2: |  | ||||||
| + добавлено поле CPU clock |  | ||||||
| + добавлена функция для обновления сетевого интерфейса (100*time_interval) |  | ||||||
| * изменено обновление слайдеров |  | ||||||
|  |  | ||||||
| Вер.1.1.1: |  | ||||||
| + добавлены тэги $swapmb, $memmb, $netdev, @@netdev=...@@ |  | ||||||
| - небольшие фиксы |  | ||||||
| * виджет теперь состоит из нескольких полей |  | ||||||
| * отредактирован конфигурационный интерфейс |  | ||||||
|  |  | ||||||
| Вер.1.0: |  | ||||||
| Первый релиз |  | ||||||
| @ -6,8 +6,8 @@ for more details. To avoid manual labor there is automatic cmake target named | |||||||
| `clangformat` (see below). Some additional detail see below. | `clangformat` (see below). Some additional detail see below. | ||||||
|  |  | ||||||
| * Indent is only spaces. 4 spaces. | * Indent is only spaces. 4 spaces. | ||||||
| * It is highly recommended to name private variables with `m_` prefix (`m_foo`). | * Any private variable should start with `m_` prefix (`m_foo`). The only one | ||||||
|   There is no exceptions for properties. |   exception is `Ui` object which should be named as `ui`. | ||||||
| * Avoid to create a large methods. Exception: if method contains lambda functions. | * Avoid to create a large methods. Exception: if method contains lambda functions. | ||||||
| * If some method is called only once, it is recommended to use lambda functions. | * If some method is called only once, it is recommended to use lambda functions. | ||||||
|   Exception is `Q_INVOKABLE` methods. |   Exception is `Q_INVOKABLE` methods. | ||||||
| @ -36,24 +36,18 @@ for more details. To avoid manual labor there is automatic cmake target named | |||||||
|   ``` |   ``` | ||||||
|  |  | ||||||
| * `Q_PROPERTY` macro is allowed and recommended for QObject based classes. | * `Q_PROPERTY` macro is allowed and recommended for QObject based classes. | ||||||
| * Qt macros (e.g. `signals`, `slots`, `Q_OBJECT`, etc) are allowed. | * Qt macros (e.g. `signals`, `slots`, `Q_OBJECT`, etc) are allowed. In other hand | ||||||
| * Current project standard is **C++11**. | `Q_FOREACH` (`foreach`) is not allowed use `for (auto &foo : bar)` instead. | ||||||
|  | * Current project standard is **C++17**. | ||||||
| * Do not use C-like code: | * Do not use C-like code: | ||||||
|     * C-like style iteration if possible. Use `Q_FOREACH` (`foreach`) and |     * C-like style iteration if possible. Use `for (auto &foo : bar)` and | ||||||
|       `std::for_each` instead if possible. It is also recommended to use iterators. |       `std::for_each` instead if possible. It is also recommended to use iterators. | ||||||
|     * C-like casts, use `const_cast`, `static_cast`, `dymanic_Cast` instead. Using |     * C-like casts, use `const_cast`, `static_cast`, `dymanic_Cast` instead. Using | ||||||
|       of `reinterpret_cast` is not recommended. It is highly recommended to use |       of `reinterpret_cast` is not recommended. It is highly recommended to use | ||||||
|       `dynamic_Cast` with the exception catching. It is also possible to use |       `dynamic_cast` with the exception catching. It is also possible to use | ||||||
|       `qvariant_cast` if required. Exception is class constructors, e.g.: |       `qvariant_cast` if required. | ||||||
|  |  | ||||||
|       ``` |  | ||||||
|       char c = 'c'; |  | ||||||
|       std::string s = "string"; |  | ||||||
|       qDebug() << QString("some string") << QChar(c) << QString(s); |  | ||||||
|       ``` |  | ||||||
|  |  | ||||||
|     * C-like `NULL`, use `nullptr` instead. |     * C-like `NULL`, use `nullptr` instead. | ||||||
| * It is highly recommended to avoid implicit casts. |     * C-like constant definition, use `const vartype foo = bar` definition instead. | ||||||
| * Abstract classes (which have at least one pure virtual method) are allowed. | * Abstract classes (which have at least one pure virtual method) are allowed. | ||||||
| * Templates are allowed and recommended. Templates usually should be described | * Templates are allowed and recommended. Templates usually should be described | ||||||
|   inside header not source code file. |   inside header not source code file. | ||||||
| @ -81,6 +75,10 @@ for more details. To avoid manual labor there is automatic cmake target named | |||||||
| * Create one file (source and header) per class. | * Create one file (source and header) per class. | ||||||
| * `else if` construction is allowed and recommended. | * `else if` construction is allowed and recommended. | ||||||
| * 'true ? foo : bar' construction is allowed and recommended for one-line assignment. | * 'true ? foo : bar' construction is allowed and recommended for one-line assignment. | ||||||
|  | * Any global pointer should be assign to `nullptr` after deletion and before | ||||||
|  |   initialization. Exception: if object is deleted into class destructor. | ||||||
|  | * Do not use semicolon in qml files unless it is required. | ||||||
|  | * Any method argument including class constructors should start with `_`. | ||||||
|  |  | ||||||
| Comments | Comments | ||||||
| -------- | -------- | ||||||
| @ -101,7 +99,7 @@ Development | |||||||
|  |  | ||||||
| * Officially the latest libraries versions should be used. In addition it is | * Officially the latest libraries versions should be used. In addition it is | ||||||
|   possible to add workarounds for all versions (usually by using preprocessor |   possible to add workarounds for all versions (usually by using preprocessor | ||||||
|   directives). |   directives); in this case patches should be placed to `packages` directory. | ||||||
| * Build should not contain any warning. | * Build should not contain any warning. | ||||||
| * Try to minimize message in Release build with logging disabled. It is highly | * Try to minimize message in Release build with logging disabled. It is highly | ||||||
|   recommended to fix KDE/Qt specific warning if possible |   recommended to fix KDE/Qt specific warning if possible | ||||||
| @ -128,6 +126,8 @@ HIG | |||||||
| --- | --- | ||||||
|  |  | ||||||
| The recommended HIG is [KDE one](https://techbase.kde.org/Projects/Usability/HIG). | The recommended HIG is [KDE one](https://techbase.kde.org/Projects/Usability/HIG). | ||||||
|  | Avoid to paint interfaces inside plugin because QML and C++ parts may have different | ||||||
|  | theming. | ||||||
|  |  | ||||||
| Licensing | Licensing | ||||||
| --------- | --------- | ||||||
| @ -142,13 +142,12 @@ For logging please use [QLoggingCategory](http://doc.qt.io/qt-5/qloggingcategory | |||||||
| Available categories should be declared in `awdebug.*` files. The following log | Available categories should be declared in `awdebug.*` files. The following log | ||||||
| levels should be used: | levels should be used: | ||||||
|  |  | ||||||
| * **debug** (`qCDebug()`) - method arguments information. | * **debug** (`qCDebug()`) - method arguments information. Please note that it | ||||||
|  |   is recommended to logging all arguments in the one line. | ||||||
| * **info** (`qCInfo()`) - additional information inside methods. | * **info** (`qCInfo()`) - additional information inside methods. | ||||||
| * **warning** (`qCWarning()`) - not critical information, which may be caused by | * **warning** (`qCWarning()`) - not critical information, which may be caused by | ||||||
|   mistakes in configuration for example. |   mistakes in configuration for example. | ||||||
| * **error** (`qCError()`) - an error which has been captured in runtime. All errors | * **critical** (`qCCritical()`) - a critical error. After this error program may | ||||||
|   should have own callback methods. |  | ||||||
| * **critical** (`qCCritical()`) - a critical error. After this error program will |  | ||||||
|   be terminated. |   be terminated. | ||||||
|  |  | ||||||
| The empty log string (e.g. `qCDebug();`) is not allowed because the method names | The empty log string (e.g. `qCDebug();`) is not allowed because the method names | ||||||
| @ -165,8 +164,7 @@ Testing | |||||||
|     1. `-DCMAKE_BUILD_TYPE=Debug`. |     1. `-DCMAKE_BUILD_TYPE=Debug`. | ||||||
|     2. `-DCMAKE_BUILD_TYPE=Debug -DBUILD_TESTING=ON`. |     2. `-DCMAKE_BUILD_TYPE=Debug -DBUILD_TESTING=ON`. | ||||||
|     3. `-DCMAKE_BUILD_TYPE=Release`. |     3. `-DCMAKE_BUILD_TYPE=Release`. | ||||||
| * It is recommended to create addition test if possible. | * Additional test functions should be declated and used only inside `BUILD_TESTING` | ||||||
| * Addition test functions should be declated and used only inside `BUILD_TESTING` |  | ||||||
|   definition. |   definition. | ||||||
|  |  | ||||||
| Tools | Tools | ||||||
| @ -197,9 +195,9 @@ Tools | |||||||
| * Recommended class constructor for QObject based classes: | * Recommended class constructor for QObject based classes: | ||||||
|  |  | ||||||
|   ``` |   ``` | ||||||
|   FooClass::FooClass(QObject *parent, const QVariant var) |   FooClass::FooClass(QObject *_parent, const QVariant _var) | ||||||
|       : QObject(parent) |       : QObject(_parent) | ||||||
|       , m_var(var) |       , m_var(_var) | ||||||
|   { |   { | ||||||
|       qCDebug(LOG_AW) << __PRETTY_FUNCTION__; |       qCDebug(LOG_AW) << __PRETTY_FUNCTION__; | ||||||
|       // some code below if any |       // some code below if any | ||||||
|  | |||||||
							
								
								
									
										15
									
								
								README.md
									
									
									
									
									
								
							
							
						
						
									
										15
									
								
								README.md
									
									
									
									
									
								
							| @ -1,6 +1,9 @@ | |||||||
| awesome-widgets (ex-pytextmonitor) | awesome-widgets (ex-pytextmonitor) | ||||||
| ================================== | ================================== | ||||||
|  |  | ||||||
|  | [](https://github.com/arcan1s/awesome-widgets/actions/workflows/build.yml) | ||||||
|  | [](https://scan.coverity.com/projects/awesome-widgets) | ||||||
|  |  | ||||||
| Information | Information | ||||||
| ----------- | ----------- | ||||||
|  |  | ||||||
| @ -28,6 +31,7 @@ Dependencies | |||||||
| ------------ | ------------ | ||||||
|  |  | ||||||
| * plasma-framework | * plasma-framework | ||||||
|  | * ksysguard (since plasma 5.22) | ||||||
|  |  | ||||||
| Optional dependencies | Optional dependencies | ||||||
| --------------------- | --------------------- | ||||||
| @ -36,6 +40,7 @@ Optional dependencies | |||||||
| * hddtemp | * hddtemp | ||||||
| * smartmontools | * smartmontools | ||||||
| * music player (mpd or MPRIS supported) | * music player (mpd or MPRIS supported) | ||||||
|  | * wireless_tools | ||||||
|  |  | ||||||
| Make dependencies | Make dependencies | ||||||
| ----------------- | ----------------- | ||||||
| @ -43,6 +48,8 @@ Make dependencies | |||||||
| * cmake | * cmake | ||||||
| * extra-cmake-modules | * extra-cmake-modules | ||||||
|  |  | ||||||
|  | In addition some distros might require to install some -dev packages, e.g. the list of required packages for deb-based distros can be found [here](https://github.com/arcan1s/awesome-widgets/blob/development/.docker/Dockerfile-ubuntu-amd64#L7). | ||||||
|  |  | ||||||
| Installation | Installation | ||||||
| ------------ | ------------ | ||||||
|  |  | ||||||
| @ -50,7 +57,7 @@ Installation | |||||||
| * install | * install | ||||||
|  |  | ||||||
|         mkdir build && cd build |         mkdir build && cd build | ||||||
|         cmake -DCMAKE_BUILD_TYPE=Release -DCMAKE_INSTALL_PREFIX=/usr ../ |         cmake -DCMAKE_BUILD_TYPE=Release -DCMAKE_INSTALL_PREFIX=/usr ../sources | ||||||
|         make && sudo make install |         make && sudo make install | ||||||
|  |  | ||||||
|   **NOTE** on Plasma 5 it very likely requires `-DKDE_INSTALL_USE_QT_SYS_PATHS=ON` flag |   **NOTE** on Plasma 5 it very likely requires `-DKDE_INSTALL_USE_QT_SYS_PATHS=ON` flag | ||||||
| @ -66,9 +73,9 @@ See [milestones](https://github.com/arcan1s/awesome-widgets/milestones) for more | |||||||
| Links | Links | ||||||
| ----- | ----- | ||||||
|  |  | ||||||
| * [Homepage](http://arcanis.name/projects/awesome-widgets/) | * [Homepage](https://arcanis.me/projects/awesome-widgets/) | ||||||
| * [Migration to 2.*](http://arcanis.name/en/2014/09/04/migration-to-v2/) | * [Migration to 2.*](https://arcanis.me/en/2014/09/04/migration-to-v2/) | ||||||
| * [Scripts and bars](http://arcanis.name/en/2014/12/19/aw-v21-bells-and-whistles/) | * [Scripts and bars](https://arcanis.me/en/2014/12/19/aw-v21-bells-and-whistles/) | ||||||
| * Plasmoid on [kde-look](http://kde-look.org/content/show.php/Awesome+Widgets?content=157124) | * Plasmoid on [kde-look](http://kde-look.org/content/show.php/Awesome+Widgets?content=157124) | ||||||
| * DataEngine on [kde-look](http://kde-look.org/content/show.php/Extended+Systemmonitor+DataEngine?content=158773) | * DataEngine on [kde-look](http://kde-look.org/content/show.php/Extended+Systemmonitor+DataEngine?content=158773) | ||||||
| * Archlinux [AUR](https://aur.archlinux.org/packages/plasma5-applet-awesome-widgets/) package | * Archlinux [AUR](https://aur.archlinux.org/packages/plasma5-applet-awesome-widgets/) package | ||||||
|  | |||||||
| @ -1,32 +1,20 @@ | |||||||
| #!/bin/bash | #!/bin/bash | ||||||
|  |  | ||||||
| SRCDIR="sources" | SRCDIR="sources" | ||||||
| MAJOR=$(grep -m1 PROJECT_VERSION_MAJOR sources/CMakeLists.txt | awk '{print $2}' | sed 's/^.\(.*\)..$/\1/') | VERSION="$(git describe --tags --abbrev=0)" | ||||||
| MINOR=$(grep -m1 PROJECT_VERSION_MINOR sources/CMakeLists.txt | awk '{print $2}' | sed 's/^.\(.*\)..$/\1/') |  | ||||||
| PATCH=$(grep -m1 PROJECT_VERSION_PATCH sources/CMakeLists.txt | awk '{print $2}' | sed 's/^.\(.*\)..$/\1/') |  | ||||||
| VERSION="${MAJOR}.${MINOR}.${PATCH}" |  | ||||||
|  |  | ||||||
| # update submodules |  | ||||||
| git submodule update --init --recursive |  | ||||||
|  |  | ||||||
| # build widget | # build widget | ||||||
| ARCHIVE="awesome-widgets" | ARCHIVE="awesome-widgets" | ||||||
| FILES="AUTHORS CHANGELOG CHANGELOG-RU COPYING" | FILES="AUTHORS CHANGELOG COPYING packages patches sources" | ||||||
| IGNORELIST="build usr .kdev4 *.kdev4 .idea" | IGNORELIST="build usr .kdev4 *.kdev4 .idea packages/*src.tar.xz" | ||||||
| # create archive | # create archive | ||||||
| [[ -e ${ARCHIVE}-${VERSION}-src.tar.xz ]] && rm -f "${ARCHIVE}-${VERSION}-src.tar.xz" | [[ -e ${ARCHIVE}-${VERSION}-src.tar.xz ]] && rm -f "${ARCHIVE}-${VERSION}-src.tar.xz" | ||||||
| [[ -d ${ARCHIVE} ]] && rm -rf "${ARCHIVE}" | [[ -d ${ARCHIVE} ]] && rm -rf "${ARCHIVE}" | ||||||
|  |  | ||||||
| cp -r "${SRCDIR}" "${ARCHIVE}" | cp -r "${SRCDIR}" "${ARCHIVE}" | ||||||
| for FILE in ${FILES[*]}; do cp -r "$FILE" "${ARCHIVE}"; done | for FILE in ${FILES[*]}; do cp -r "$FILE" "${ARCHIVE}"; done | ||||||
| for FILE in ${IGNORELIST[*]}; do rm -rf "${ARCHIVE}/${FILE}"; done | for FILE in ${IGNORELIST[*]}; do rm -rf "${ARCHIVE}/${FILE}"; done | ||||||
| tar cJf "${ARCHIVE}-${VERSION}-src.tar.xz" "${ARCHIVE}" |  | ||||||
| ln -sf "../${ARCHIVE}-${VERSION}-src.tar.xz" packages |  | ||||||
| rm -rf "${ARCHIVE}" |  | ||||||
|  |  | ||||||
| # update md5sum | tar cJf "${ARCHIVE}-${VERSION}-src.tar.xz" "${ARCHIVE}" | ||||||
| MD5SUMS=$(md5sum ${ARCHIVE}-${VERSION}-src.tar.xz | awk '{print $1}') |  | ||||||
| sed -i "/md5sums=('[0-9A-Fa-f]*/s/[^'][^)]*/md5sums=('${MD5SUMS}'/" packages/PKGBUILD | rm -rf "${ARCHIVE}" | ||||||
| sed -i "s/pkgver=[0-9.]*/pkgver=${VERSION}/" packages/PKGBUILD |  | ||||||
| # clear |  | ||||||
| find . -type f -name '*src.tar.xz' -not -name "*${VERSION}-src.tar.xz" -exec rm -rf {} \; |  | ||||||
| find packages -type l -xtype l -exec rm -rf {} \; |  | ||||||
|  | |||||||
| @ -2,22 +2,22 @@ | |||||||
|  |  | ||||||
| pkgname=plasma5-applet-awesome-widgets | pkgname=plasma5-applet-awesome-widgets | ||||||
| _pkgname=awesome-widgets | _pkgname=awesome-widgets | ||||||
| pkgver=3.0.0 | pkgver=3.4.3 | ||||||
| pkgrel=1 | pkgrel=1 | ||||||
| pkgdesc="Collection of minimalistic Plasmoids which look like Awesome WM widgets (ex-PyTextMonitor)" | pkgdesc="Collection of minimalistic Plasmoids which look like Awesome WM widgets (ex-PyTextMonitor)" | ||||||
| arch=('i686' 'x86_64') | arch=('i686' 'x86_64') | ||||||
| url="http://arcanis.name/projects/awesome-widgets" | url="https://arcanis.me/projects/awesome-widgets" | ||||||
| license=('GPL3') | license=('GPL3') | ||||||
| depends=('plasma-framework') | depends=('ksysguard' 'plasma-framework') | ||||||
| optdepends=("catalyst: for GPU monitor" | optdepends=("catalyst: for GPU monitor" | ||||||
|             "hddtemp: for HDD temperature monitor" |             "hddtemp: for HDD temperature monitor" | ||||||
|             "smartmontools: for HDD temperature monitor" |             "smartmontools: for HDD temperature monitor" | ||||||
|             "mpd: for music player monitor" |             "mpd: for music player monitor" | ||||||
|             "nvidia-utils: for GPU monitor") |             "nvidia-utils: for GPU monitor") | ||||||
| makedepends=('cmake' 'extra-cmake-modules') | makedepends=('cmake' 'extra-cmake-modules' 'python') | ||||||
| source=(https://github.com/arcan1s/awesome-widgets/releases/download/V.${pkgver}/${_pkgname}-${pkgver}-src.tar.xz) | source=(https://github.com/arcan1s/awesome-widgets/releases/download/V.${pkgver}/${_pkgname}-${pkgver}-src.tar.xz) | ||||||
| install=${pkgname}.install | install=${pkgname}.install | ||||||
| md5sums=('4878899f092ee12ad8f935f0551ef6b2') | md5sums=('5953ba518191bb6fff83cdb8633c735c') | ||||||
| backup=('etc/xdg/plasma-dataengine-extsysmon.conf') | backup=('etc/xdg/plasma-dataengine-extsysmon.conf') | ||||||
|  |  | ||||||
| prepare() { | prepare() { | ||||||
| @ -28,8 +28,9 @@ prepare() { | |||||||
| build () { | build () { | ||||||
|   cd "${srcdir}/build" |   cd "${srcdir}/build" | ||||||
|   cmake -DKDE_INSTALL_USE_QT_SYS_PATHS=ON \ |   cmake -DKDE_INSTALL_USE_QT_SYS_PATHS=ON \ | ||||||
|         -DCMAKE_BUILD_TYPE=Release \ |         -DCMAKE_BUILD_TYPE=Optimization \ | ||||||
|         -DCMAKE_INSTALL_PREFIX=/usr \ |         -DCMAKE_INSTALL_PREFIX=/usr \ | ||||||
|  |         -DBUILD_FUTURE=ON \ | ||||||
|         "../${_pkgname}" |         "../${_pkgname}" | ||||||
|   make |   make | ||||||
| } | } | ||||||
|  | |||||||
| @ -6,7 +6,7 @@ pkgver=2.2.1.r15.g78931b3 | |||||||
| pkgrel=1 | pkgrel=1 | ||||||
| pkgdesc="Collection of minimalistic Plasmoids which look like Awesome WM widgets (ex-PyTextMonitor). Git version" | pkgdesc="Collection of minimalistic Plasmoids which look like Awesome WM widgets (ex-PyTextMonitor). Git version" | ||||||
| arch=('i686' 'x86_64') | arch=('i686' 'x86_64') | ||||||
| url="http://arcanis.name/projects/awesome-widgets" | url="https://arcanis.me/projects/awesome-widgets" | ||||||
| license=('GPL3') | license=('GPL3') | ||||||
| depends=('plasma-framework') | depends=('plasma-framework') | ||||||
| optdepends=("amarok: for music player monitor" | optdepends=("amarok: for music player monitor" | ||||||
|  | |||||||
| @ -1,11 +0,0 @@ | |||||||
| cmake |  | ||||||
| extra-cmake-modules |  | ||||||
| g++ |  | ||||||
| git |  | ||||||
| libkf5i18n-dev |  | ||||||
| libkf5notifications-dev |  | ||||||
| libkf5service-dev |  | ||||||
| libkf5windowsystem-dev |  | ||||||
| plasma-framework-dev |  | ||||||
| qtbase5-dev |  | ||||||
| qtdeclarative5-dev |  | ||||||
							
								
								
									
										7
									
								
								patches/PATCHING.md
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										7
									
								
								patches/PATCHING.md
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,7 @@ | |||||||
|  | * Using git tree from root git directory: | ||||||
|  |  | ||||||
|  |         git apply path/to/patch | ||||||
|  |  | ||||||
|  | * Using release tarball from root directory: | ||||||
|  |  | ||||||
|  |         patch -p2 -i path/to/patch | ||||||
| @ -1,27 +1,26 @@ | |||||||
| --- | --- | ||||||
| Language:        Cpp | Language:        Cpp | ||||||
| AccessModifierOffset: -4 | AccessModifierOffset: -4 | ||||||
| AlignAfterOpenBracket: true | AlignAfterOpenBracket: Align | ||||||
| AlignConsecutiveAssignments: false | AlignConsecutiveAssignments: None | ||||||
| AlignEscapedNewlinesLeft: false |  | ||||||
| AlignOperands:   true | AlignOperands:   true | ||||||
| AlignTrailingComments: true | AlignTrailingComments: true | ||||||
| AllowAllParametersOfDeclarationOnNextLine: true | AllowAllParametersOfDeclarationOnNextLine: true | ||||||
| AllowShortBlocksOnASingleLine: false | AllowShortBlocksOnASingleLine: Never | ||||||
| AllowShortCaseLabelsOnASingleLine: false | AllowShortCaseLabelsOnASingleLine: false | ||||||
| AllowShortFunctionsOnASingleLine: Inline | AllowShortFunctionsOnASingleLine: Inline | ||||||
| AllowShortIfStatementsOnASingleLine: false | AllowShortIfStatementsOnASingleLine: Never | ||||||
| AllowShortLoopsOnASingleLine: false | AllowShortLoopsOnASingleLine: false | ||||||
| AlwaysBreakAfterDefinitionReturnType: None | AlwaysBreakAfterDefinitionReturnType: None | ||||||
| AlwaysBreakBeforeMultilineStrings: false | AlwaysBreakBeforeMultilineStrings: false | ||||||
| AlwaysBreakTemplateDeclarations: false | AlwaysBreakTemplateDeclarations: No | ||||||
| BinPackArguments: true | BinPackArguments: true | ||||||
| BinPackParameters: true | BinPackParameters: true | ||||||
| BreakBeforeBinaryOperators: All | BreakBeforeBinaryOperators: All | ||||||
| BreakBeforeBraces: Linux | BreakBeforeBraces: Linux | ||||||
| BreakBeforeTernaryOperators: true | BreakBeforeTernaryOperators: true | ||||||
| BreakConstructorInitializersBeforeComma: true | BreakConstructorInitializersBeforeComma: true | ||||||
| ColumnLimit:     80 | ColumnLimit:     120 | ||||||
| CommentPragmas:  '^ IWYU pragma:' | CommentPragmas:  '^ IWYU pragma:' | ||||||
| ConstructorInitializerAllOnOneLineOrOnePerLine: false | ConstructorInitializerAllOnOneLineOrOnePerLine: false | ||||||
| ConstructorInitializerIndentWidth: 4 | ConstructorInitializerIndentWidth: 4 | ||||||
| @ -59,7 +58,7 @@ SpacesInContainerLiterals: true | |||||||
| SpacesInCStyleCastParentheses: false | SpacesInCStyleCastParentheses: false | ||||||
| SpacesInParentheses: false | SpacesInParentheses: false | ||||||
| SpacesInSquareBrackets: false | SpacesInSquareBrackets: false | ||||||
| Standard:        Cpp11 | Standard:        Latest | ||||||
| TabWidth:        8 | TabWidth:        8 | ||||||
| UseTab:          Never | UseTab:          Never | ||||||
| ... | ... | ||||||
|  | |||||||
| @ -1,26 +0,0 @@ | |||||||
| [Buildset] |  | ||||||
| BuildItems=@Variant(\x00\x00\x00\t\x00\x00\x00\x00\x01\x00\x00\x00\x0b\x00\x00\x00\x00\x01\x00\x00\x00\x1e\x00a\x00w\x00e\x00s\x00o\x00m\x00e\x00-\x00w\x00i\x00d\x00g\x00e\x00t\x00s) |  | ||||||
|  |  | ||||||
| [CMake] |  | ||||||
| Build Directory Count=1 |  | ||||||
| Current Build Directory Index=0 |  | ||||||
| ProjectRootRelative=./ |  | ||||||
|  |  | ||||||
| [CMake][CMake Build Directory 0] |  | ||||||
| Build Directory Path=file:///home/arcanis/Documents/github/awesome-widgets/build |  | ||||||
| Build Type=Optimization |  | ||||||
| CMake Binary=file:///usr/bin/cmake |  | ||||||
| Environment Profile= |  | ||||||
| Extra Arguments= |  | ||||||
| Install Directory=file:///usr |  | ||||||
|  |  | ||||||
| [Defines And Includes][Compiler] |  | ||||||
| Name=GCC |  | ||||||
| Path=gcc |  | ||||||
| Type=GCC |  | ||||||
|  |  | ||||||
| [Launch] |  | ||||||
| Launch Configurations= |  | ||||||
|  |  | ||||||
| [Project] |  | ||||||
| VersionControlSupport=kdevgit |  | ||||||
| @ -1,23 +0,0 @@ | |||||||
| [Buildset] |  | ||||||
| BuildItems=@Variant(\x00\x00\x00\t\x00\x00\x00\x00\x01\x00\x00\x00\x0b\x00\x00\x00\x00\x01\x00\x00\x00\x1e\x00a\x00w\x00e\x00s\x00o\x00m\x00e\x00-\x00w\x00i\x00d\x00g\x00e\x00t\x00s) |  | ||||||
|  |  | ||||||
| [CMake] |  | ||||||
| Build Directory Count=1 |  | ||||||
| Current Build Directory Index=0 |  | ||||||
| ProjectRootRelative=./ |  | ||||||
|  |  | ||||||
| [CMake][CMake Build Directory 0] |  | ||||||
| Build Directory Path=file:///home/arcanis/Documents/github/awesome-widgets/build |  | ||||||
| Build Type=Release |  | ||||||
| CMake Binary=file:///usr/bin/cmake |  | ||||||
| Environment Profile= |  | ||||||
| Extra Arguments= |  | ||||||
| Install Directory=file:///usr |  | ||||||
|  |  | ||||||
| [Defines And Includes][Compiler] |  | ||||||
| Name=GCC |  | ||||||
| Path=gcc |  | ||||||
| Type=GCC |  | ||||||
|  |  | ||||||
| [Project] |  | ||||||
| VersionControlSupport=kdevgit |  | ||||||
							
								
								
									
										1
									
								
								sources/3rdparty/fontdialog
									
									
									
									
										vendored
									
									
								
							
							
								
								
								
								
								
							
						
						
									
										1
									
								
								sources/3rdparty/fontdialog
									
									
									
									
										vendored
									
									
								
							 Submodule sources/3rdparty/fontdialog deleted from ef47b1146e
									
								
							
							
								
								
									
										165
									
								
								sources/3rdparty/fontdialog/COPYING
									
									
									
									
										vendored
									
									
										Normal file
									
								
							
							
						
						
									
										165
									
								
								sources/3rdparty/fontdialog/COPYING
									
									
									
									
										vendored
									
									
										Normal file
									
								
							| @ -0,0 +1,165 @@ | |||||||
|  | GNU LESSER GENERAL PUBLIC LICENSE | ||||||
|  |                        Version 3, 29 June 2007 | ||||||
|  |  | ||||||
|  |  Copyright (C) 2007 Free Software Foundation, Inc. <http://fsf.org/> | ||||||
|  |  Everyone is permitted to copy and distribute verbatim copies | ||||||
|  |  of this license document, but changing it is not allowed. | ||||||
|  |  | ||||||
|  |  | ||||||
|  |   This version of the GNU Lesser General Public License incorporates | ||||||
|  | the terms and conditions of version 3 of the GNU General Public | ||||||
|  | License, supplemented by the additional permissions listed below. | ||||||
|  |  | ||||||
|  |   0. Additional Definitions. | ||||||
|  |  | ||||||
|  |   As used herein, "this License" refers to version 3 of the GNU Lesser | ||||||
|  | General Public License, and the "GNU GPL" refers to version 3 of the GNU | ||||||
|  | General Public License. | ||||||
|  |  | ||||||
|  |   "The Library" refers to a covered work governed by this License, | ||||||
|  | other than an Application or a Combined Work as defined below. | ||||||
|  |  | ||||||
|  |   An "Application" is any work that makes use of an interface provided | ||||||
|  | by the Library, but which is not otherwise based on the Library. | ||||||
|  | Defining a subclass of a class defined by the Library is deemed a mode | ||||||
|  | of using an interface provided by the Library. | ||||||
|  |  | ||||||
|  |   A "Combined Work" is a work produced by combining or linking an | ||||||
|  | Application with the Library.  The particular version of the Library | ||||||
|  | with which the Combined Work was made is also called the "Linked | ||||||
|  | Version". | ||||||
|  |  | ||||||
|  |   The "Minimal Corresponding Source" for a Combined Work means the | ||||||
|  | Corresponding Source for the Combined Work, excluding any source code | ||||||
|  | for portions of the Combined Work that, considered in isolation, are | ||||||
|  | based on the Application, and not on the Linked Version. | ||||||
|  |  | ||||||
|  |   The "Corresponding Application Code" for a Combined Work means the | ||||||
|  | object code and/or source code for the Application, including any data | ||||||
|  | and utility programs needed for reproducing the Combined Work from the | ||||||
|  | Application, but excluding the System Libraries of the Combined Work. | ||||||
|  |  | ||||||
|  |   1. Exception to Section 3 of the GNU GPL. | ||||||
|  |  | ||||||
|  |   You may convey a covered work under sections 3 and 4 of this License | ||||||
|  | without being bound by section 3 of the GNU GPL. | ||||||
|  |  | ||||||
|  |   2. Conveying Modified Versions. | ||||||
|  |  | ||||||
|  |   If you modify a copy of the Library, and, in your modifications, a | ||||||
|  | facility refers to a function or data to be supplied by an Application | ||||||
|  | that uses the facility (other than as an argument passed when the | ||||||
|  | facility is invoked), then you may convey a copy of the modified | ||||||
|  | version: | ||||||
|  |  | ||||||
|  |    a) under this License, provided that you make a good faith effort to | ||||||
|  |    ensure that, in the event an Application does not supply the | ||||||
|  |    function or data, the facility still operates, and performs | ||||||
|  |    whatever part of its purpose remains meaningful, or | ||||||
|  |  | ||||||
|  |    b) under the GNU GPL, with none of the additional permissions of | ||||||
|  |    this License applicable to that copy. | ||||||
|  |  | ||||||
|  |   3. Object Code Incorporating Material from Library Header Files. | ||||||
|  |  | ||||||
|  |   The object code form of an Application may incorporate material from | ||||||
|  | a header file that is part of the Library.  You may convey such object | ||||||
|  | code under terms of your choice, provided that, if the incorporated | ||||||
|  | material is not limited to numerical parameters, data structure | ||||||
|  | layouts and accessors, or small macros, inline functions and templates | ||||||
|  | (ten or fewer lines in length), you do both of the following: | ||||||
|  |  | ||||||
|  |    a) Give prominent notice with each copy of the object code that the | ||||||
|  |    Library is used in it and that the Library and its use are | ||||||
|  |    covered by this License. | ||||||
|  |  | ||||||
|  |    b) Accompany the object code with a copy of the GNU GPL and this license | ||||||
|  |    document. | ||||||
|  |  | ||||||
|  |   4. Combined Works. | ||||||
|  |  | ||||||
|  |   You may convey a Combined Work under terms of your choice that, | ||||||
|  | taken together, effectively do not restrict modification of the | ||||||
|  | portions of the Library contained in the Combined Work and reverse | ||||||
|  | engineering for debugging such modifications, if you also do each of | ||||||
|  | the following: | ||||||
|  |  | ||||||
|  |    a) Give prominent notice with each copy of the Combined Work that | ||||||
|  |    the Library is used in it and that the Library and its use are | ||||||
|  |    covered by this License. | ||||||
|  |  | ||||||
|  |    b) Accompany the Combined Work with a copy of the GNU GPL and this license | ||||||
|  |    document. | ||||||
|  |  | ||||||
|  |    c) For a Combined Work that displays copyright notices during | ||||||
|  |    execution, include the copyright notice for the Library among | ||||||
|  |    these notices, as well as a reference directing the user to the | ||||||
|  |    copies of the GNU GPL and this license document. | ||||||
|  |  | ||||||
|  |    d) Do one of the following: | ||||||
|  |  | ||||||
|  |        0) Convey the Minimal Corresponding Source under the terms of this | ||||||
|  |        License, and the Corresponding Application Code in a form | ||||||
|  |        suitable for, and under terms that permit, the user to | ||||||
|  |        recombine or relink the Application with a modified version of | ||||||
|  |        the Linked Version to produce a modified Combined Work, in the | ||||||
|  |        manner specified by section 6 of the GNU GPL for conveying | ||||||
|  |        Corresponding Source. | ||||||
|  |  | ||||||
|  |        1) Use a suitable shared library mechanism for linking with the | ||||||
|  |        Library.  A suitable mechanism is one that (a) uses at run time | ||||||
|  |        a copy of the Library already present on the user's computer | ||||||
|  |        system, and (b) will operate properly with a modified version | ||||||
|  |        of the Library that is interface-compatible with the Linked | ||||||
|  |        Version. | ||||||
|  |  | ||||||
|  |    e) Provide Installation Information, but only if you would otherwise | ||||||
|  |    be required to provide such information under section 6 of the | ||||||
|  |    GNU GPL, and only to the extent that such information is | ||||||
|  |    necessary to install and execute a modified version of the | ||||||
|  |    Combined Work produced by recombining or relinking the | ||||||
|  |    Application with a modified version of the Linked Version. (If | ||||||
|  |    you use option 4d0, the Installation Information must accompany | ||||||
|  |    the Minimal Corresponding Source and Corresponding Application | ||||||
|  |    Code. If you use option 4d1, you must provide the Installation | ||||||
|  |    Information in the manner specified by section 6 of the GNU GPL | ||||||
|  |    for conveying Corresponding Source.) | ||||||
|  |  | ||||||
|  |   5. Combined Libraries. | ||||||
|  |  | ||||||
|  |   You may place library facilities that are a work based on the | ||||||
|  | Library side by side in a single library together with other library | ||||||
|  | facilities that are not Applications and are not covered by this | ||||||
|  | License, and convey such a combined library under terms of your | ||||||
|  | choice, if you do both of the following: | ||||||
|  |  | ||||||
|  |    a) Accompany the combined library with a copy of the same work based | ||||||
|  |    on the Library, uncombined with any other library facilities, | ||||||
|  |    conveyed under the terms of this License. | ||||||
|  |  | ||||||
|  |    b) Give prominent notice with the combined library that part of it | ||||||
|  |    is a work based on the Library, and explaining where to find the | ||||||
|  |    accompanying uncombined form of the same work. | ||||||
|  |  | ||||||
|  |   6. Revised Versions of the GNU Lesser General Public License. | ||||||
|  |  | ||||||
|  |   The Free Software Foundation may publish revised and/or new versions | ||||||
|  | of the GNU Lesser General Public License from time to time. Such new | ||||||
|  | versions will be similar in spirit to the present version, but may | ||||||
|  | differ in detail to address new problems or concerns. | ||||||
|  |  | ||||||
|  |   Each version is given a distinguishing version number. If the | ||||||
|  | Library as you received it specifies that a certain numbered version | ||||||
|  | of the GNU Lesser General Public License "or any later version" | ||||||
|  | applies to it, you have the option of following the terms and | ||||||
|  | conditions either of that published version or of any later version | ||||||
|  | published by the Free Software Foundation. If the Library as you | ||||||
|  | received it does not specify a version number of the GNU Lesser | ||||||
|  | General Public License, you may choose any version of the GNU Lesser | ||||||
|  | General Public License ever published by the Free Software Foundation. | ||||||
|  |  | ||||||
|  |   If the Library as you received it specifies that a proxy can decide | ||||||
|  | whether future versions of the GNU Lesser General Public License shall | ||||||
|  | apply, that proxy's public statement of acceptance of any version is | ||||||
|  | permanent authorization for you to choose that version for the | ||||||
|  | Library. | ||||||
							
								
								
									
										4
									
								
								sources/3rdparty/fontdialog/README.md
									
									
									
									
										vendored
									
									
										Normal file
									
								
							
							
						
						
									
										4
									
								
								sources/3rdparty/fontdialog/README.md
									
									
									
									
										vendored
									
									
										Normal file
									
								
							| @ -0,0 +1,4 @@ | |||||||
|  | qtadds-fontdialog | ||||||
|  | ================= | ||||||
|  |  | ||||||
|  | Font dialog which provides a font color settings | ||||||
							
								
								
									
										215
									
								
								sources/3rdparty/fontdialog/fontdialog.cpp
									
									
									
									
										vendored
									
									
										Normal file
									
								
							
							
						
						
									
										215
									
								
								sources/3rdparty/fontdialog/fontdialog.cpp
									
									
									
									
										vendored
									
									
										Normal file
									
								
							| @ -0,0 +1,215 @@ | |||||||
|  | /*************************************************************************** | ||||||
|  |  *  Copyright (C) 2014  Evgeniy Alekseev                                      * | ||||||
|  |  *                                                                         * | ||||||
|  |  *  This library is free software; you can redistribute it and/or          * | ||||||
|  |  *  modify it under the terms of the GNU Lesser General Public             * | ||||||
|  |  *  License as published by the Free Software Foundation; either           * | ||||||
|  |  *  version 3.0 of the License, or (at your option) any later version.     * | ||||||
|  |  *                                                                         * | ||||||
|  |  *  This library is distributed in the hope that it will be useful,        * | ||||||
|  |  *  but WITHOUT ANY WARRANTY; without even the implied warranty of         * | ||||||
|  |  *  MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU       * | ||||||
|  |  *  Lesser General Public License for more details.                        * | ||||||
|  |  *                                                                         * | ||||||
|  |  *  You should have received a copy of the GNU Lesser General Public       * | ||||||
|  |  *  License along with this library.                                       * | ||||||
|  |  ***************************************************************************/ | ||||||
|  |  | ||||||
|  | #include "fontdialog.h" | ||||||
|  |  | ||||||
|  | #include <QGridLayout> | ||||||
|  |  | ||||||
|  |  | ||||||
|  | CFont::CFont(const QString family, int pointSize, int weight, bool italic, QColor color) | ||||||
|  |     : QFont(family, pointSize, weight, italic) | ||||||
|  | { | ||||||
|  |     setCurrentColor(color); | ||||||
|  | } | ||||||
|  |  | ||||||
|  |  | ||||||
|  | QColor CFont::color() | ||||||
|  | { | ||||||
|  |     return currentColor; | ||||||
|  | } | ||||||
|  |  | ||||||
|  |  | ||||||
|  | void CFont::setCurrentColor(const QColor color) | ||||||
|  | { | ||||||
|  |     currentColor = color; | ||||||
|  | } | ||||||
|  |  | ||||||
|  |  | ||||||
|  | int CFont::html2QFont(const int htmlWeight) | ||||||
|  | { | ||||||
|  |     int weight = 16; | ||||||
|  |     switch(htmlWeight) { | ||||||
|  |     case 100: | ||||||
|  |         weight = 16; | ||||||
|  |         break; | ||||||
|  |     case 200: | ||||||
|  |     case 300: | ||||||
|  |         weight = 25; | ||||||
|  |         break; | ||||||
|  |     case 400: | ||||||
|  |         weight = 50; | ||||||
|  |         break; | ||||||
|  |     case 500: | ||||||
|  |     case 600: | ||||||
|  |         weight = 63; | ||||||
|  |         break; | ||||||
|  |     case 700: | ||||||
|  |     case 800: | ||||||
|  |         weight = 75; | ||||||
|  |         break; | ||||||
|  |     case 900: | ||||||
|  |         weight = 87; | ||||||
|  |         break; | ||||||
|  |     default: | ||||||
|  |         break; | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     return weight; | ||||||
|  | } | ||||||
|  |  | ||||||
|  |  | ||||||
|  | int CFont::qFont2html(const int weight) | ||||||
|  | { | ||||||
|  |     int htmlWeight = 400; | ||||||
|  |     switch(weight) { | ||||||
|  |     case 16: | ||||||
|  |         htmlWeight = 100; | ||||||
|  |         break; | ||||||
|  |     case 25: | ||||||
|  |         htmlWeight = 300; | ||||||
|  |         break; | ||||||
|  |     case 50: | ||||||
|  |         htmlWeight = 400; | ||||||
|  |         break; | ||||||
|  |     case 63: | ||||||
|  |         htmlWeight = 600; | ||||||
|  |         break; | ||||||
|  |     case 75: | ||||||
|  |         htmlWeight = 800; | ||||||
|  |         break; | ||||||
|  |     case 87: | ||||||
|  |         htmlWeight = 900; | ||||||
|  |         break; | ||||||
|  |     default: | ||||||
|  |         break; | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     return htmlWeight; | ||||||
|  | } | ||||||
|  |  | ||||||
|  |  | ||||||
|  | int CFont::htmlWeight() | ||||||
|  | { | ||||||
|  |     return CFont::qFont2html(weight()); | ||||||
|  | } | ||||||
|  |  | ||||||
|  |  | ||||||
|  | void CFont::setHtmlWeight(const int htmlWeight) | ||||||
|  | { | ||||||
|  |     setWeight(CFont::html2QFont(htmlWeight)); | ||||||
|  | } | ||||||
|  |  | ||||||
|  |  | ||||||
|  | CFont CFont::fromQFont(const QFont font, const QColor color) | ||||||
|  | { | ||||||
|  |     return CFont(font.family(), font.pointSize(), font.weight(), font.italic(), color); | ||||||
|  | } | ||||||
|  |  | ||||||
|  |  | ||||||
|  | QFont CFont::toQFont() | ||||||
|  | { | ||||||
|  |     return QFont(family(), pointSize(), weight(), italic()); | ||||||
|  | } | ||||||
|  |  | ||||||
|  |  | ||||||
|  | CFontDialog::CFontDialog(QWidget *parent, bool needWeight, bool needItalic) | ||||||
|  |     : QDialog(parent) | ||||||
|  | { | ||||||
|  |     QGridLayout *mainGrid = new QGridLayout(this); | ||||||
|  |     setLayout(mainGrid); | ||||||
|  |  | ||||||
|  |     colorBox = new QComboBox(this); | ||||||
|  |     connect(colorBox, SIGNAL(currentIndexChanged(QString)), this, SLOT(updateColor(QString))); | ||||||
|  |     QStringList colorNames = QColor::colorNames(); | ||||||
|  |     int index = 0; | ||||||
|  |     for (int i=0; i<colorNames.count(); i++) { | ||||||
|  |         QColor color(colorNames[i]); | ||||||
|  |         colorBox->addItem(colorNames[i], color); | ||||||
|  |         QModelIndex idx = colorBox->model()->index(index++, 0); | ||||||
|  |         colorBox->model()->setData(idx, color, Qt::BackgroundRole); | ||||||
|  |     } | ||||||
|  |     mainGrid->addWidget(colorBox, 0, 0); | ||||||
|  |     fontBox = new QFontComboBox(this); | ||||||
|  |     mainGrid->addWidget(fontBox, 0, 1); | ||||||
|  |     sizeBox = new QSpinBox(this); | ||||||
|  |     mainGrid->addWidget(sizeBox, 0, 2); | ||||||
|  |     weightBox = new QSpinBox(this); | ||||||
|  |     mainGrid->addWidget(weightBox, 0, 3); | ||||||
|  |     italicBox = new QComboBox(this); | ||||||
|  |     italicBox->addItem(tr("normal")); | ||||||
|  |     italicBox->addItem(tr("italic")); | ||||||
|  |     mainGrid->addWidget(italicBox, 0, 4); | ||||||
|  |  | ||||||
|  |     buttons = new QDialogButtonBox(QDialogButtonBox::Ok | QDialogButtonBox::Cancel, | ||||||
|  |                                    Qt::Horizontal, this); | ||||||
|  |     QObject::connect(buttons, SIGNAL(accepted()), this, SLOT(accept())); | ||||||
|  |     QObject::connect(buttons, SIGNAL(rejected()), this, SLOT(reject())); | ||||||
|  |     mainGrid->addWidget(buttons, 1, 0, 1, 5); | ||||||
|  |  | ||||||
|  |     italicBox->setHidden(!needItalic); | ||||||
|  |     weightBox->setHidden(!needWeight); | ||||||
|  | } | ||||||
|  |  | ||||||
|  |  | ||||||
|  | CFontDialog::~CFontDialog() | ||||||
|  | { | ||||||
|  |     delete colorBox; | ||||||
|  |     delete buttons; | ||||||
|  |     delete fontBox; | ||||||
|  |     delete italicBox; | ||||||
|  |     delete sizeBox; | ||||||
|  |     delete weightBox; | ||||||
|  | } | ||||||
|  |  | ||||||
|  |  | ||||||
|  | void CFontDialog::updateColor(const QString color) | ||||||
|  | { | ||||||
|  |     colorBox->setStyleSheet(QString("background:%1").arg(QColor(color).name())); | ||||||
|  | } | ||||||
|  |  | ||||||
|  |  | ||||||
|  | CFont CFontDialog::getFont(const QString title, CFont defaultFont, bool needWeight, bool needItalic, int *status) | ||||||
|  | { | ||||||
|  |     CFontDialog dlg(0, needWeight, needItalic); | ||||||
|  |  | ||||||
|  |     dlg.setWindowTitle(title); | ||||||
|  |     QStringList colorNames = QColor::colorNames(); | ||||||
|  |     for (int i=0; i<colorNames.count(); i++) | ||||||
|  |         if (QColor(colorNames[i]) == defaultFont.color()) { | ||||||
|  |             dlg.colorBox->setCurrentIndex(i); | ||||||
|  |             break; | ||||||
|  |         } | ||||||
|  |     dlg.fontBox->setCurrentFont(defaultFont); | ||||||
|  |     dlg.sizeBox->setValue(defaultFont.pointSize()); | ||||||
|  |     dlg.weightBox->setValue(defaultFont.weight()); | ||||||
|  |     if (defaultFont.italic()) | ||||||
|  |         dlg.italicBox->setCurrentIndex(1); | ||||||
|  |     else | ||||||
|  |         dlg.italicBox->setCurrentIndex(0); | ||||||
|  |  | ||||||
|  |     CFont font = CFont(defaultFont); | ||||||
|  |     int ret = dlg.exec(); | ||||||
|  |     if (ret == 1) | ||||||
|  |         font = CFont(dlg.fontBox->currentFont().family(), | ||||||
|  |                      dlg.sizeBox->value(), | ||||||
|  |                      dlg.weightBox->value(), | ||||||
|  |                      dlg.italicBox->currentIndex() == 1, | ||||||
|  |                      QColor(dlg.colorBox->currentText())); | ||||||
|  |     if (status != nullptr) | ||||||
|  |         *status = ret; | ||||||
|  |     return font; | ||||||
|  | } | ||||||
							
								
								
									
										82
									
								
								sources/3rdparty/fontdialog/fontdialog.h
									
									
									
									
										vendored
									
									
										Normal file
									
								
							
							
						
						
									
										82
									
								
								sources/3rdparty/fontdialog/fontdialog.h
									
									
									
									
										vendored
									
									
										Normal file
									
								
							| @ -0,0 +1,82 @@ | |||||||
|  | /*************************************************************************** | ||||||
|  |  *  Copyright (C) 2014  Evgeniy Alekseev                                      * | ||||||
|  |  *                                                                         * | ||||||
|  |  *  This library is free software; you can redistribute it and/or          * | ||||||
|  |  *  modify it under the terms of the GNU Lesser General Public             * | ||||||
|  |  *  License as published by the Free Software Foundation; either           * | ||||||
|  |  *  version 3.0 of the License, or (at your option) any later version.     * | ||||||
|  |  *                                                                         * | ||||||
|  |  *  This library is distributed in the hope that it will be useful,        * | ||||||
|  |  *  but WITHOUT ANY WARRANTY; without even the implied warranty of         * | ||||||
|  |  *  MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU       * | ||||||
|  |  *  Lesser General Public License for more details.                        * | ||||||
|  |  *                                                                         * | ||||||
|  |  *  You should have received a copy of the GNU Lesser General Public       * | ||||||
|  |  *  License along with this library.                                       * | ||||||
|  |  ***************************************************************************/ | ||||||
|  |  | ||||||
|  | #ifndef FONTDIALOG_H | ||||||
|  | #define FONTDIALOG_H | ||||||
|  |  | ||||||
|  | #include <QComboBox> | ||||||
|  | #include <QDialog> | ||||||
|  | #include <QDialogButtonBox> | ||||||
|  | #include <QFontComboBox> | ||||||
|  | #include <QSpinBox> | ||||||
|  |  | ||||||
|  |  | ||||||
|  | class CFont : public QFont | ||||||
|  | { | ||||||
|  | public: | ||||||
|  |     explicit CFont(const QString family, int pointSize = -1, | ||||||
|  |                    int weight = -1, bool italic = false, | ||||||
|  |                    QColor color = QColor(QString("#000000"))); | ||||||
|  |     // color properties | ||||||
|  |     QColor color(); | ||||||
|  |     void setCurrentColor(const QColor color); | ||||||
|  |     // html weight properties | ||||||
|  |     static int html2QFont(const int htmlWeight); | ||||||
|  |     static int qFont2html(const int weight); | ||||||
|  |     int htmlWeight(); | ||||||
|  |     void setHtmlWeight(const int htmlWeight); | ||||||
|  |     // conversion to QFont | ||||||
|  |     static CFont fromQFont(const QFont font, | ||||||
|  |                            const QColor color = QColor(QString("#000000"))); | ||||||
|  |     QFont toQFont(); | ||||||
|  |  | ||||||
|  | private: | ||||||
|  |     QColor currentColor; | ||||||
|  | }; | ||||||
|  |  | ||||||
|  |  | ||||||
|  | class CFontDialog : public QDialog | ||||||
|  | { | ||||||
|  |     Q_OBJECT | ||||||
|  |  | ||||||
|  | public: | ||||||
|  |     explicit CFontDialog(QWidget *parent = 0, | ||||||
|  |                          const bool needWeight = true, | ||||||
|  |                          const bool needItalic = true); | ||||||
|  |     ~CFontDialog(); | ||||||
|  |     static CFont getFont(const QString title = tr("Select font"), | ||||||
|  |                          CFont defaultFont = CFont(QString("Arial"), | ||||||
|  |                                                    12, 400, false, | ||||||
|  |                                                    QColor(QString("#000000"))), | ||||||
|  |                          const bool needWeight = true, | ||||||
|  |                          const bool needItalic = true, | ||||||
|  |                          int *status = nullptr); | ||||||
|  |  | ||||||
|  | private slots: | ||||||
|  |     void updateColor(const QString color); | ||||||
|  |  | ||||||
|  | private: | ||||||
|  |     QComboBox *colorBox; | ||||||
|  |     QDialogButtonBox *buttons; | ||||||
|  |     QFontComboBox *fontBox; | ||||||
|  |     QComboBox *italicBox; | ||||||
|  |     QSpinBox *sizeBox; | ||||||
|  |     QSpinBox *weightBox; | ||||||
|  | }; | ||||||
|  |  | ||||||
|  |  | ||||||
|  | #endif /* FONTDIALOG_H */ | ||||||
| @ -12,7 +12,7 @@ QReplyTimeout::QReplyTimeout(QNetworkReply *reply, const int timeout) | |||||||
|  |  | ||||||
| void QReplyTimeout::timeout() | void QReplyTimeout::timeout() | ||||||
| { | { | ||||||
|     QNetworkReply *reply = static_cast<QNetworkReply *>(parent()); |     auto *reply = dynamic_cast<QNetworkReply *>(parent()); | ||||||
|     if (reply->isRunning()) |     if (reply->isRunning()) | ||||||
|         reply->close(); |         reply->close(); | ||||||
| } | } | ||||||
|  | |||||||
							
								
								
									
										1
									
								
								sources/3rdparty/task
									
									
									
									
										vendored
									
									
								
							
							
								
								
								
								
								
							
						
						
									
										1
									
								
								sources/3rdparty/task
									
									
									
									
										vendored
									
									
								
							 Submodule sources/3rdparty/task deleted from d2798204a1
									
								
							
							
								
								
									
										1
									
								
								sources/3rdparty/tasks
									
									
									
									
										vendored
									
									
								
							
							
								
								
								
								
								
							
						
						
									
										1
									
								
								sources/3rdparty/tasks
									
									
									
									
										vendored
									
									
								
							 Submodule sources/3rdparty/tasks deleted from 230bdecd2c
									
								
							| @ -1,12 +1,13 @@ | |||||||
| cmake_minimum_required(VERSION 2.8.12) | cmake_minimum_required(VERSION 3.16.0) | ||||||
|  |  | ||||||
| # some fucking magic | # some fucking magic | ||||||
| cmake_policy(SET CMP0003 OLD) |  | ||||||
| cmake_policy(SET CMP0002 OLD) |  | ||||||
| cmake_policy(SET CMP0011 NEW) | cmake_policy(SET CMP0011 NEW) | ||||||
| cmake_policy(SET CMP0015 NEW) | cmake_policy(SET CMP0015 NEW) | ||||||
| if (POLICY CMP0063) | if (POLICY CMP0063) | ||||||
|     cmake_policy(SET CMP0063 OLD) |     cmake_policy(SET CMP0063 NEW) | ||||||
|  | endif () | ||||||
|  | if (POLICY CMP0071) | ||||||
|  |     cmake_policy(SET CMP0071 NEW) | ||||||
| endif () | endif () | ||||||
|  |  | ||||||
| project(awesomewidgets) | project(awesomewidgets) | ||||||
| @ -14,7 +15,7 @@ set(PROJECT_AUTHOR "Evgeniy Alekseev") | |||||||
| set(PROJECT_CONTACT "esalexeev@gmail.com") | set(PROJECT_CONTACT "esalexeev@gmail.com") | ||||||
| set(PROJECT_LICENSE "GPL3") | set(PROJECT_LICENSE "GPL3") | ||||||
| set(PROJECT_VERSION_MAJOR "3") | set(PROJECT_VERSION_MAJOR "3") | ||||||
| set(PROJECT_VERSION_MINOR "0") | set(PROJECT_VERSION_MINOR "5") | ||||||
| set(PROJECT_VERSION_PATCH "0") | set(PROJECT_VERSION_PATCH "0") | ||||||
| set(PROJECT_VERSION "${PROJECT_VERSION_MAJOR}.${PROJECT_VERSION_MINOR}.${PROJECT_VERSION_PATCH}") | set(PROJECT_VERSION "${PROJECT_VERSION_MAJOR}.${PROJECT_VERSION_MINOR}.${PROJECT_VERSION_PATCH}") | ||||||
| # append git version if any | # append git version if any | ||||||
| @ -34,49 +35,42 @@ option(BUILD_DEB_PACKAGE "Build deb package" OFF) | |||||||
| option(BUILD_RPM_PACKAGE "Build rpm package" OFF) | option(BUILD_RPM_PACKAGE "Build rpm package" OFF) | ||||||
| # build details | # build details | ||||||
| option(BUILD_FUTURE "Build with the features which will be marked as stable later" OFF) | option(BUILD_FUTURE "Build with the features which will be marked as stable later" OFF) | ||||||
|  | option(BUILD_LOAD "Build with additional load" OFF) | ||||||
| option(BUILD_TESTING "Build with additional test abilities" OFF) | option(BUILD_TESTING "Build with additional test abilities" OFF) | ||||||
| # some additional targets |  | ||||||
| set(CLANGFORMAT_EXECUTABLE "/usr/bin/clang-format" CACHE STRING "Path to clang-format executable") |  | ||||||
| set(CPPCHECK_EXECUTABLE "/usr/bin/cppcheck" CACHE STRING "Path to cppcheck executable") |  | ||||||
|  |  | ||||||
| # flags | # generate changelog | ||||||
| if (CMAKE_COMPILER_IS_GNUCXX) | set(PROJECT_CHANGELOG "Changelog" CACHE INTERNAL "") | ||||||
|     set(CMAKE_CXX_FLAGS "-Wall -Wno-cpp -std=c++11") | include(changelog.cmake) | ||||||
|     set(CMAKE_CXX_FLAGS_DEBUG "-g -O0") |  | ||||||
|     set(CMAKE_CXX_FLAGS_RELEASE "-O2 -DNDEBUG") |  | ||||||
|     set(CMAKE_CXX_FLAGS_OPTIMIZATION "-Ofast -DNDEBUG") |  | ||||||
|     # avoid newer gcc warnings |  | ||||||
|     add_definitions(-D_DEFAULT_SOURCE) |  | ||||||
| elseif (CMAKE_CXX_COMPILER_ID MATCHES "Clang") |  | ||||||
|     set(CMAKE_CXX_FLAGS "-Wall -std=c++11 -stdlib=libc++") |  | ||||||
|     set(CMAKE_CXX_FLAGS_DEBUG "-g -O0") |  | ||||||
|     set(CMAKE_CXX_FLAGS_RELEASE "-O2 -DNDEBUG") |  | ||||||
|     set(CMAKE_CXX_FLAGS_OPTIMIZATION "-Ofast -DNDEBUG") |  | ||||||
|     # linker flags |  | ||||||
|     set(CMAKE_EXE_LINKER_FLAGS "-lc++abi") |  | ||||||
|     set(CMAKE_MODULE_LINKER_FLAGS "-lc++abi") |  | ||||||
|     set(CMAKE_SHARED_LINKER_FLAGS "-lc++abi") |  | ||||||
| else () |  | ||||||
|     message(FATAL_ERROR "Unknown compiler") |  | ||||||
| endif () |  | ||||||
| if (CMAKE_BUILD_TYPE MATCHES Debug) |  | ||||||
|     set(CMAKE_VERBOSE_MAKEFILE ON) |  | ||||||
| endif () |  | ||||||
|  |  | ||||||
|  | # directories | ||||||
| set(PROJECT_TRDPARTY_DIR ${CMAKE_CURRENT_SOURCE_DIR}/3rdparty) | set(PROJECT_TRDPARTY_DIR ${CMAKE_CURRENT_SOURCE_DIR}/3rdparty) | ||||||
| set(PROJECT_LIBRARY awesomewidgets) | set(PROJECT_LIBRARY awesomewidgets) | ||||||
|  | set(PROJECT_MONITORSOURCES extsysmonsources) | ||||||
|  |  | ||||||
|  | # modules | ||||||
|  | include(compiler.cmake) | ||||||
| include(libraries.cmake) | include(libraries.cmake) | ||||||
| include(clang-format.cmake) | include(clang-format.cmake) | ||||||
| include(cppcheck.cmake) | include(cppcheck.cmake) | ||||||
|  | include(coverity.cmake) | ||||||
|  |  | ||||||
|  | # pre-configure | ||||||
| get_directory_property(CMAKE_DEFINITIONS COMPILE_DEFINITIONS) | get_directory_property(CMAKE_DEFINITIONS COMPILE_DEFINITIONS) | ||||||
| configure_file(${CMAKE_SOURCE_DIR}/version.h.in ${CMAKE_CURRENT_BINARY_DIR}/version.h) | configure_file(${CMAKE_SOURCE_DIR}/version.h.in ${CMAKE_CURRENT_BINARY_DIR}/version.h) | ||||||
|  |  | ||||||
|  | # build components | ||||||
| add_subdirectory(awesomewidgets) | add_subdirectory(awesomewidgets) | ||||||
|  | add_subdirectory(extsysmonsources) | ||||||
| add_subdirectory(extsysmon) | add_subdirectory(extsysmon) | ||||||
| if (BUILD_PLASMOIDS) | if (BUILD_PLASMOIDS) | ||||||
|  |     add_subdirectory(qml) | ||||||
|     add_subdirectory(awesome-widget) |     add_subdirectory(awesome-widget) | ||||||
|     add_subdirectory(desktop-panel) |     add_subdirectory(desktop-panel) | ||||||
|     add_subdirectory(translations) |     add_subdirectory(translations) | ||||||
| endif () | endif () | ||||||
|  | if (BUILD_TESTING) | ||||||
|  |     enable_testing() | ||||||
|  |     add_subdirectory(test) | ||||||
|  | endif () | ||||||
|  |  | ||||||
| include(packages-recipe.cmake) | include(packages-recipe.cmake) | ||||||
|  | |||||||
| @ -16,69 +16,115 @@ | |||||||
|  ***************************************************************************/ |  ***************************************************************************/ | ||||||
|  |  | ||||||
|  |  | ||||||
|  | #include <KI18n/KLocalizedString> | ||||||
|  |  | ||||||
| #include "awdebug.h" | #include "awdebug.h" | ||||||
| #include "version.h" |  | ||||||
|  |  | ||||||
|  |  | ||||||
| Q_LOGGING_CATEGORY(LOG_AW, "org.kde.plasma.awesomewidget", | Q_LOGGING_CATEGORY(LOG_AW, "org.kde.plasma.awesomewidget", QtMsgType::QtWarningMsg) | ||||||
|                    QtMsgType::QtWarningMsg) | Q_LOGGING_CATEGORY(LOG_DBUS, "org.kde.plasma.awdbus", QtMsgType::QtWarningMsg) | ||||||
| Q_LOGGING_CATEGORY(LOG_DP, "org.kde.plasma.desktoppanel", | Q_LOGGING_CATEGORY(LOG_DP, "org.kde.plasma.desktoppanel", QtMsgType::QtWarningMsg) | ||||||
|                    QtMsgType::QtWarningMsg) |  | ||||||
| Q_LOGGING_CATEGORY(LOG_ESM, "org.kde.plasma.extsysmon", QtMsgType::QtWarningMsg) | Q_LOGGING_CATEGORY(LOG_ESM, "org.kde.plasma.extsysmon", QtMsgType::QtWarningMsg) | ||||||
| Q_LOGGING_CATEGORY(LOG_LIB, "org.kde.plasma.awesomewidgets", | Q_LOGGING_CATEGORY(LOG_ESS, "org.kde.plasma.extsysmonsources", QtMsgType::QtWarningMsg) | ||||||
|                    QtMsgType::QtWarningMsg) | Q_LOGGING_CATEGORY(LOG_LIB, "org.kde.plasma.awesomewidgets", QtMsgType::QtWarningMsg) | ||||||
|  |  | ||||||
|  |  | ||||||
| const QStringList getBuildData() | QString AWDebug::getAboutText(const QString &_type) | ||||||
|  | { | ||||||
|  |     QString text; | ||||||
|  |     if (_type == "header") { | ||||||
|  |         text = NAME; | ||||||
|  |     } else if (_type == "version") { | ||||||
|  |         text = i18n("Version %1 (build date %2)", VERSION, BUILD_DATE); | ||||||
|  |         if (!QString(COMMIT_SHA).isEmpty()) | ||||||
|  |             text += QString(" (%1)").arg(COMMIT_SHA); | ||||||
|  |     } else if (_type == "description") { | ||||||
|  |         text = i18n("A set of minimalistic plasmoid widgets"); | ||||||
|  |     } else if (_type == "links") { | ||||||
|  |         text = i18n("Links:") + "<ul>" + QString("<li><a href=\"%1\">%2</a></li>").arg(HOMEPAGE).arg(i18n("Homepage")) | ||||||
|  |                + QString("<li><a href=\"%1\">%2</a></li>").arg(REPOSITORY).arg(i18n("Repository")) | ||||||
|  |                + QString("<li><a href=\"%1\">%2</a></li>").arg(BUGTRACKER).arg(i18n("Bugtracker")) | ||||||
|  |                + QString("<li><a href=\"%1\">%2</a></li>").arg(TRANSLATION).arg(i18n("Translation issue")) | ||||||
|  |                + QString("<li><a href=\"%1\">%2</a></li>").arg(AUR_PACKAGES).arg(i18n("AUR packages")) | ||||||
|  |                + QString("<li><a href=\"%1\">%2</a></li>").arg(OPENSUSE_PACKAGES).arg(i18n("openSUSE packages")) | ||||||
|  |                + "</ul>"; | ||||||
|  |     } else if (_type == "copy") { | ||||||
|  |         text = QString("<small>© %1 <a href=\"mailto:%2\">%3</a><br>").arg(DATE).arg(EMAIL).arg(AUTHOR) | ||||||
|  |                + i18nc("This software is licensed under %1", LICENSE) + "</small>"; | ||||||
|  |     } else if (_type == "translators") { | ||||||
|  |         QStringList translatorList = QString(TRANSLATORS).split(','); | ||||||
|  |         for (auto &translator : translatorList) | ||||||
|  |             translator = QString("<li>%1</li>").arg(translator); | ||||||
|  |         text = i18n("Translators:") + "<ul>" + translatorList.join("") + "</ul>"; | ||||||
|  |     } else if (_type == "3rdparty") { | ||||||
|  |         QStringList trdPartyList = QString(TRDPARTY_LICENSE).split(';', Qt::SkipEmptyParts); | ||||||
|  |         for (int i = 0; i < trdPartyList.count(); i++) | ||||||
|  |             trdPartyList[i] = QString("<li><a href=\"%3\">%1</a> (%2 license)</li>") | ||||||
|  |                                   .arg(trdPartyList.at(i).split(',')[0]) | ||||||
|  |                                   .arg(trdPartyList.at(i).split(',')[1]) | ||||||
|  |                                   .arg(trdPartyList.at(i).split(',')[2]); | ||||||
|  |         text = i18n("This software uses:") + "<ul>" + trdPartyList.join("") + "</ul>"; | ||||||
|  |     } else if (_type == "thanks") { | ||||||
|  |         QStringList thanks = QString(SPECIAL_THANKS).split(';', Qt::SkipEmptyParts); | ||||||
|  |         for (int i = 0; i < thanks.count(); i++) | ||||||
|  |             thanks[i] = QString("<li><a href=\"%2\">%1</a></li>") | ||||||
|  |                             .arg(thanks.at(i).split(',')[0]) | ||||||
|  |                             .arg(thanks.at(i).split(',')[1]); | ||||||
|  |         text = i18n("Special thanks to:") + "<ul>" + thanks.join("") + "</ul>"; | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     return text; | ||||||
|  | } | ||||||
|  |  | ||||||
|  |  | ||||||
|  | QStringList AWDebug::getBuildData() | ||||||
| { | { | ||||||
|     QStringList metadata; |     QStringList metadata; | ||||||
|     metadata.append(QString("=== Awesome Widgets configuration details ===")); |     metadata.append("=== Awesome Widgets configuration details ==="); | ||||||
|     // build information |     // build information | ||||||
|     metadata.append(QString("Build details:")); |     metadata.append("Build details:"); | ||||||
|     metadata.append(QString("    VERSION: %1").arg(VERSION)); |     metadata.append(QString("    VERSION: %1").arg(VERSION)); | ||||||
|     metadata.append(QString("    COMMIT_SHA: %1").arg(COMMIT_SHA)); |     metadata.append(QString("    COMMIT_SHA: %1").arg(COMMIT_SHA)); | ||||||
|     metadata.append(QString("    BUILD_DATE: %1").arg(BUILD_DATE)); |     metadata.append(QString("    BUILD_DATE: %1").arg(BUILD_DATE)); | ||||||
|     // configuration |     // configuration | ||||||
|     metadata.append(QString("API details:")); |     metadata.append("API details:"); | ||||||
|     metadata.append(QString("    AWGIAPI: %1").arg(AWGIAPI)); |     metadata.append(QString("    AW_GRAPHITEM_API: %1").arg(AW_GRAPHITEM_API)); | ||||||
|     metadata.append(QString("    AWEQAPI: %1").arg(AWEQAPI)); |     metadata.append(QString("    AW_EXTQUOTES_API: %1").arg(AW_EXTQUOTES_API)); | ||||||
|     metadata.append(QString("    AWESAPI: %1").arg(AWESAPI)); |     metadata.append(QString("    AW_EXTSCRIPT_API: %1").arg(AW_EXTSCRIPT_API)); | ||||||
|     metadata.append(QString("    AWEUAPI: %1").arg(AWEUAPI)); |     metadata.append(QString("    AW_EXTUPGRADE_API: %1").arg(AW_EXTUPGRADE_API)); | ||||||
|     metadata.append(QString("    AWEWAPI: %1").arg(AWEWAPI)); |     metadata.append(QString("    AW_EXTWEATHER_API: %1").arg(AW_EXTWEATHER_API)); | ||||||
|  |     metadata.append(QString("    AW_FORMATTER_API: %1").arg(AW_FORMATTER_API)); | ||||||
|  |     metadata.append(QString("    REQUEST_TIMEOUT: %1").arg(REQUEST_TIMEOUT)); | ||||||
|     metadata.append(QString("    TIME_KEYS: %1").arg(TIME_KEYS)); |     metadata.append(QString("    TIME_KEYS: %1").arg(TIME_KEYS)); | ||||||
|  |     metadata.append(QString("    STATIC_KEYS: %1").arg(STATIC_KEYS)); | ||||||
|     // cmake properties |     // cmake properties | ||||||
|     metadata.append(QString("cmake properties:")); |     metadata.append("cmake properties:"); | ||||||
|     metadata.append(QString("    CMAKE_BUILD_TYPE: %1").arg(CMAKE_BUILD_TYPE)); |     metadata.append(QString("    CMAKE_BUILD_TYPE: %1").arg(CMAKE_BUILD_TYPE)); | ||||||
|     metadata.append( |     metadata.append(QString("    CMAKE_CXX_COMPILER: %1").arg(CMAKE_CXX_COMPILER)); | ||||||
|         QString("    CMAKE_CXX_COMPILER: %1").arg(CMAKE_CXX_COMPILER)); |  | ||||||
|     metadata.append(QString("    CMAKE_CXX_FLAGS: %1").arg(CMAKE_CXX_FLAGS)); |     metadata.append(QString("    CMAKE_CXX_FLAGS: %1").arg(CMAKE_CXX_FLAGS)); | ||||||
|     metadata.append( |     metadata.append(QString("    CMAKE_CXX_FLAGS_DEBUG: %1").arg(CMAKE_CXX_FLAGS_DEBUG)); | ||||||
|         QString("    CMAKE_CXX_FLAGS_DEBUG: %1").arg(CMAKE_CXX_FLAGS_DEBUG)); |     metadata.append(QString("    CMAKE_CXX_FLAGS_RELEASE: %1").arg(CMAKE_CXX_FLAGS_RELEASE)); | ||||||
|     metadata.append(QString("    CMAKE_CXX_FLAGS_RELEASE: %1") |     metadata.append(QString("    CMAKE_CXX_FLAGS_OPTIMIZATION: %1").arg(CMAKE_CXX_FLAGS_OPTIMIZATION)); | ||||||
|                         .arg(CMAKE_CXX_FLAGS_RELEASE)); |     metadata.append(QString("    CMAKE_DEFINITIONS: %1").arg(CMAKE_DEFINITIONS)); | ||||||
|     metadata.append(QString("    CMAKE_CXX_FLAGS_OPTIMIZATION: %1") |     metadata.append(QString("    CMAKE_INSTALL_PREFIX: %1").arg(CMAKE_INSTALL_PREFIX)); | ||||||
|                         .arg(CMAKE_CXX_FLAGS_OPTIMIZATION)); |     metadata.append(QString("    CMAKE_MODULE_LINKER_FLAGS: %1").arg(CMAKE_MODULE_LINKER_FLAGS)); | ||||||
|     metadata.append( |     metadata.append(QString("    CMAKE_SHARED_LINKER_FLAGS: %1").arg(CMAKE_SHARED_LINKER_FLAGS)); | ||||||
|         QString("    CMAKE_DEFINITIONS: %1").arg(CMAKE_DEFINITIONS)); |  | ||||||
|     metadata.append( |  | ||||||
|         QString("    CMAKE_INSTALL_PREFIX: %1").arg(CMAKE_INSTALL_PREFIX)); |  | ||||||
|     metadata.append(QString("    CMAKE_MODULE_LINKER_FLAGS: %1") |  | ||||||
|                         .arg(CMAKE_MODULE_LINKER_FLAGS)); |  | ||||||
|     metadata.append(QString("    CMAKE_SHARED_LINKER_FLAGS: %1") |  | ||||||
|                         .arg(CMAKE_SHARED_LINKER_FLAGS)); |  | ||||||
|     // components |     // components | ||||||
|     metadata.append(QString("Components data:")); |     metadata.append("Components data:"); | ||||||
|     metadata.append(QString("    BUILD_PLASMOIDS: %1").arg(BUILD_PLASMOIDS)); |     metadata.append(QString("    BUILD_PLASMOIDS: %1").arg(BUILD_PLASMOIDS)); | ||||||
|     metadata.append( |     metadata.append(QString("    BUILD_DEB_PACKAGE: %1").arg(BUILD_DEB_PACKAGE)); | ||||||
|         QString("    BUILD_DEB_PACKAGE: %1").arg(BUILD_DEB_PACKAGE)); |     metadata.append(QString("    BUILD_RPM_PACKAGE: %1").arg(BUILD_RPM_PACKAGE)); | ||||||
|     metadata.append( |     metadata.append(QString("    CLANGFORMAT_EXECUTABLE: %1").arg(CLANGFORMAT_EXECUTABLE)); | ||||||
|         QString("    BUILD_RPM_PACKAGE: %1").arg(BUILD_RPM_PACKAGE)); |     metadata.append(QString("    COVERITY_COMMENT: %1").arg(COVERITY_COMMENT)); | ||||||
|     metadata.append( |     metadata.append(QString("    COVERITY_DIRECTORY: %1").arg(COVERITY_DIRECTORY)); | ||||||
|         QString("    CLANGFORMAT_EXECUTABLE: %1").arg(CLANGFORMAT_EXECUTABLE)); |     metadata.append(QString("    COVERITY_EMAIL: %1").arg(COVERITY_EMAIL)); | ||||||
|     metadata.append( |     metadata.append(QString("    COVERITY_EXECUTABLE: %1").arg(COVERITY_EXECUTABLE)); | ||||||
|         QString("    CPPCHECK_EXECUTABLE: %1").arg(CPPCHECK_EXECUTABLE)); |     metadata.append(QString("    COVERITY_URL: %1").arg(COVERITY_URL)); | ||||||
|     metadata.append(QString("    PROP_FUTURE: %1").arg(PROP_FUTURE)); |     metadata.append(QString("    CPPCHECK_EXECUTABLE: %1").arg(CPPCHECK_EXECUTABLE)); | ||||||
|  |     // additional functions | ||||||
|     metadata.append(QString("    PROP_FUTURE: %1").arg(PROP_FUTURE)); |     metadata.append(QString("    PROP_FUTURE: %1").arg(PROP_FUTURE)); | ||||||
|  |     metadata.append(QString("    PROP_LOAD: %1").arg(PROP_LOAD)); | ||||||
|  |     metadata.append(QString("    PROP_TEST: %1").arg(PROP_TEST)); | ||||||
|  |  | ||||||
|     return metadata; |     return metadata; | ||||||
| } | } | ||||||
|  | |||||||
| @ -21,30 +21,27 @@ | |||||||
|  |  | ||||||
| #include <QLoggingCategory> | #include <QLoggingCategory> | ||||||
|  |  | ||||||
| #ifndef LOG_FORMAT | #include "version.h" | ||||||
| #define LOG_FORMAT                                                             \ |  | ||||||
|     "[%{time process}][%{if-debug}DD%{endif}%{if-info}II%{endif}%{if-"         \ |  | ||||||
|     "warning}WW%{endif}%{if-critical}CC%{endif}%{if-fatal}FF%{endif}][%{"      \ |  | ||||||
|     "category}][%{function}] %{message}" |  | ||||||
| #endif /* LOG_FORMAT */ |  | ||||||
|  |  | ||||||
| // define info log level | namespace AWDebug | ||||||
| // #if QT_VERSION < QT_VERSION_CHECK(5, 5, 0) | { | ||||||
| // #ifndef qCInfo | const char LOG_FORMAT[] = "[%{time " | ||||||
| // redefine info because it doesn't log properly |                           "process}][%{if-debug}DD%{endif}%{if-info}II%{endif}%" | ||||||
| #ifdef qCInfo |                           "{if-warning}WW%{endif}%{if-critical}CC%{endif}%{if-" | ||||||
| #undef qCInfo |                           "fatal}FF%{endif}][%{category}][%{function}] " | ||||||
| #endif /* qCInfo */ |                           "%{message}"; | ||||||
| #define qCInfo qCDebug |  | ||||||
| // #endif /* QT_VERSION */ | QString getAboutText(const QString &_type); | ||||||
|  | QStringList getBuildData(); | ||||||
|  | } // namespace AWDebug | ||||||
|  |  | ||||||
|  |  | ||||||
| Q_DECLARE_LOGGING_CATEGORY(LOG_AW) | Q_DECLARE_LOGGING_CATEGORY(LOG_AW) | ||||||
|  | Q_DECLARE_LOGGING_CATEGORY(LOG_DBUS) | ||||||
| Q_DECLARE_LOGGING_CATEGORY(LOG_DP) | Q_DECLARE_LOGGING_CATEGORY(LOG_DP) | ||||||
| Q_DECLARE_LOGGING_CATEGORY(LOG_ESM) | Q_DECLARE_LOGGING_CATEGORY(LOG_ESM) | ||||||
|  | Q_DECLARE_LOGGING_CATEGORY(LOG_ESS) | ||||||
| Q_DECLARE_LOGGING_CATEGORY(LOG_LIB) | Q_DECLARE_LOGGING_CATEGORY(LOG_LIB) | ||||||
|  |  | ||||||
| const QStringList getBuildData(); |  | ||||||
|  |  | ||||||
|  |  | ||||||
| #endif /* AWDEBUG_H */ | #endif /* AWDEBUG_H */ | ||||||
|  | |||||||
| @ -15,14 +15,12 @@ Icon=utilities-system-monitor | |||||||
| X-KDE-ServiceTypes=Plasma/Applet | X-KDE-ServiceTypes=Plasma/Applet | ||||||
| X-Plasma-API=declarativeappletscript | X-Plasma-API=declarativeappletscript | ||||||
| X-Plasma-MainScript=ui/main.qml | X-Plasma-MainScript=ui/main.qml | ||||||
| X-Plasma-RemoteLocation= |  | ||||||
|  |  | ||||||
| X-KDE-PluginInfo-Author=Evgeniy Alekseev aka arcanis | X-KDE-PluginInfo-Author=Evgeniy Alekseev aka arcanis | ||||||
| X-KDE-PluginInfo-Email=esalexeev@gmail.com | X-KDE-PluginInfo-Email=esalexeev@gmail.com | ||||||
| X-KDE-PluginInfo-Name=org.kde.plasma.awesomewidget | X-KDE-PluginInfo-Name=org.kde.plasma.awesomewidget | ||||||
| X-KDE-PluginInfo-Version=@PROJECT_VERSION@ | X-KDE-PluginInfo-Version=@PROJECT_VERSION@ | ||||||
| X-KDE-PluginInfo-Website=http://arcanis.name/projects/awesome-widgets/ | X-KDE-PluginInfo-Website=https://arcanis.me/projects/awesome-widgets/ | ||||||
| X-KDE-PluginInfo-Category=System Information | X-KDE-PluginInfo-Category=System Information | ||||||
| X-KDE-PluginInfo-Depends= |  | ||||||
| X-KDE-PluginInfo-License=GPLv3 | X-KDE-PluginInfo-License=GPLv3 | ||||||
| X-KDE-PluginInfo-EnabledByDefault=true | X-KDE-PluginInfo-EnabledByDefault=true | ||||||
|  | |||||||
| @ -32,6 +32,9 @@ | |||||||
|         <entry name="checkUpdates" type="bool"> |         <entry name="checkUpdates" type="bool"> | ||||||
|             <default>true</default> |             <default>true</default> | ||||||
|         </entry> |         </entry> | ||||||
|  |         <entry name="optimize" type="bool"> | ||||||
|  |             <default>true</default> | ||||||
|  |         </entry> | ||||||
|         <entry name="height" type="int"> |         <entry name="height" type="int"> | ||||||
|             <default>0</default> |             <default>0</default> | ||||||
|         </entry> |         </entry> | ||||||
| @ -59,6 +62,15 @@ | |||||||
|         <entry name="acOffline" type="string"> |         <entry name="acOffline" type="string"> | ||||||
|             <default>( )</default> |             <default>( )</default> | ||||||
|         </entry> |         </entry> | ||||||
|  |         <entry name="telemetryCount" type="int"> | ||||||
|  |             <default>100</default> | ||||||
|  |         </entry> | ||||||
|  |         <entry name="telemetryRemote" type="bool"> | ||||||
|  |             <default>false</default> | ||||||
|  |         </entry> | ||||||
|  |         <entry name="telemetryId" type="string"> | ||||||
|  |             <default></default> | ||||||
|  |         </entry> | ||||||
|     </group> |     </group> | ||||||
|  |  | ||||||
|     <group name="Tooltip"> |     <group name="Tooltip"> | ||||||
| @ -95,13 +107,13 @@ | |||||||
|         <entry name="swapTooltipColor" type="string"> |         <entry name="swapTooltipColor" type="string"> | ||||||
|             <default>#ffff00</default> |             <default>#ffff00</default> | ||||||
|         </entry> |         </entry> | ||||||
|         <entry name="downTooltip" type="bool"> |         <entry name="downkbTooltip" type="bool"> | ||||||
|             <default>true</default> |             <default>true</default> | ||||||
|         </entry> |         </entry> | ||||||
|         <entry name="downTooltipColor" type="string"> |         <entry name="downkbTooltipColor" type="string"> | ||||||
|             <default>#00ffff</default> |             <default>#00ffff</default> | ||||||
|         </entry> |         </entry> | ||||||
|         <entry name="upTooltipColor" type="string"> |         <entry name="upkbTooltipColor" type="string"> | ||||||
|             <default>#ff00ff</default> |             <default>#ff00ff</default> | ||||||
|         </entry> |         </entry> | ||||||
|         <entry name="batTooltip" type="bool"> |         <entry name="batTooltip" type="bool"> | ||||||
| @ -135,6 +147,12 @@ | |||||||
|         <entry name="fontStyle" type="string"> |         <entry name="fontStyle" type="string"> | ||||||
|             <default>normal</default> |             <default>normal</default> | ||||||
|         </entry> |         </entry> | ||||||
|  |         <entry name="textStyle" type="string"> | ||||||
|  |             <default>normal</default> | ||||||
|  |         </entry> | ||||||
|  |         <entry name="textStyleColor" type="string"> | ||||||
|  |             <default>#000000</default> | ||||||
|  |         </entry> | ||||||
|     </group> |     </group> | ||||||
|  |  | ||||||
| </kcfg> | </kcfg> | ||||||
|  | |||||||
| @ -16,103 +16,26 @@ | |||||||
|  ***************************************************************************/ |  ***************************************************************************/ | ||||||
|  |  | ||||||
| import QtQuick 2.0 | import QtQuick 2.0 | ||||||
| import QtQuick.Controls 1.3 as QtControls |  | ||||||
| import QtQuick.Layouts 1.0 as QtLayouts |  | ||||||
|  |  | ||||||
| import org.kde.plasma.private.awesomewidget 1.0 | import org.kde.plasma.private.awesomewidget 1.0 | ||||||
|  |  | ||||||
|  |  | ||||||
| Item { | Item { | ||||||
|     id: aboutPage |     id: aboutPage | ||||||
|  |     // backend | ||||||
|     AWActions { |     AWActions { | ||||||
|         id: awActions |         id: awActions | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     width: childrenRect.width |     width: childrenRect.width | ||||||
|     height: childrenRect.height |     height: childrenRect.height | ||||||
|     implicitWidth: pageColumn.implicitWidth |  | ||||||
|     implicitHeight: pageColumn.implicitHeight |  | ||||||
|  |  | ||||||
|     property bool debug: awActions.isDebugEnabled() |     property bool debug: awActions.isDebugEnabled() | ||||||
|  |  | ||||||
|  |     AboutTab { | ||||||
|     Column { |         textProvider: awActions | ||||||
|         id: pageColumn |  | ||||||
|         anchors.fill: parent |  | ||||||
|         QtControls.TabView { |  | ||||||
|             height: parent.height |  | ||||||
|             width: parent.width |  | ||||||
|             QtControls.Tab { |  | ||||||
|                 anchors.margins: 10.0 |  | ||||||
|                 title: i18n("About") |  | ||||||
|  |  | ||||||
|                 QtLayouts.ColumnLayout { |  | ||||||
|                     QtControls.Label { |  | ||||||
|                         QtLayouts.Layout.fillWidth: true |  | ||||||
|                         horizontalAlignment: Text.AlignHCenter |  | ||||||
|                         text: awActions.getAboutText("header") |  | ||||||
|     } |     } | ||||||
|  |  | ||||||
|                     QtControls.Label { |  | ||||||
|                         QtLayouts.Layout.fillWidth: true |  | ||||||
|                         horizontalAlignment: Text.AlignHCenter |  | ||||||
|                         text: awActions.getAboutText("version") |  | ||||||
|                     } |  | ||||||
|  |  | ||||||
|                     QtControls.Label { |  | ||||||
|                         QtLayouts.Layout.fillWidth: true |  | ||||||
|                         horizontalAlignment: Text.AlignJustify |  | ||||||
|                         text: awActions.getAboutText("description") |  | ||||||
|                     } |  | ||||||
|  |  | ||||||
|                     QtControls.Label { |  | ||||||
|                         QtLayouts.Layout.fillWidth: true |  | ||||||
|                         horizontalAlignment: Text.AlignLeft |  | ||||||
|                         textFormat: Text.RichText |  | ||||||
|                         text: awActions.getAboutText("links") |  | ||||||
|                         onLinkActivated: Qt.openUrlExternally(link); |  | ||||||
|                     } |  | ||||||
|  |  | ||||||
|                     QtControls.Label { |  | ||||||
|                         QtLayouts.Layout.fillHeight: true |  | ||||||
|                         QtLayouts.Layout.fillWidth: true |  | ||||||
|                         font.capitalization: Font.SmallCaps |  | ||||||
|                         horizontalAlignment: Text.AlignHCenter |  | ||||||
|                         verticalAlignment: Text.AlignBottom |  | ||||||
|                         textFormat: Text.RichText |  | ||||||
|                         text: awActions.getAboutText("copy") |  | ||||||
|                     } |  | ||||||
|                 } |  | ||||||
|             } |  | ||||||
|  |  | ||||||
|             QtControls.Tab { |  | ||||||
|                 anchors.margins: 10.0 |  | ||||||
|                 title: i18n("Acknowledgment") |  | ||||||
|  |  | ||||||
|                 QtLayouts.ColumnLayout { |  | ||||||
|                     QtControls.Label { |  | ||||||
|                         QtLayouts.Layout.fillWidth: true |  | ||||||
|                         wrapMode: Text.WordWrap |  | ||||||
|                         horizontalAlignment: Text.AlignJustify |  | ||||||
|                         text: awActions.getAboutText("translators") |  | ||||||
|                     } |  | ||||||
|  |  | ||||||
|                     QtControls.Label { |  | ||||||
|                         QtLayouts.Layout.fillHeight: true |  | ||||||
|                         QtLayouts.Layout.fillWidth: true |  | ||||||
|                         wrapMode: Text.WordWrap |  | ||||||
|                         horizontalAlignment: Text.AlignJustify |  | ||||||
|                         verticalAlignment: Text.AlignTop |  | ||||||
|                         textFormat: Text.RichText |  | ||||||
|                         text: awActions.getAboutText("3rdparty") |  | ||||||
|                         onLinkActivated: Qt.openUrlExternally(link); |  | ||||||
|                     } |  | ||||||
|                 } |  | ||||||
|             } |  | ||||||
|         } |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|  |  | ||||||
|     Component.onCompleted: { |     Component.onCompleted: { | ||||||
|         if (debug) console.debug() |         if (debug) console.debug() | ||||||
|     } |     } | ||||||
|  | |||||||
| @ -27,9 +27,6 @@ Item { | |||||||
|     AWActions { |     AWActions { | ||||||
|         id: awActions |         id: awActions | ||||||
|     } |     } | ||||||
|     AWConfigHelper { |  | ||||||
|         id: awConfig |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     width: childrenRect.width |     width: childrenRect.width | ||||||
|     height: childrenRect.height |     height: childrenRect.height | ||||||
| @ -44,198 +41,98 @@ Item { | |||||||
|     property alias cfg_wrapText: wordWrap.checked |     property alias cfg_wrapText: wordWrap.checked | ||||||
|     property alias cfg_notify: notify.checked |     property alias cfg_notify: notify.checked | ||||||
|     property alias cfg_checkUpdates: updates.checked |     property alias cfg_checkUpdates: updates.checked | ||||||
|  |     property alias cfg_optimize: optimize.checked | ||||||
|     property alias cfg_height: widgetHeight.value |     property alias cfg_height: widgetHeight.value | ||||||
|     property alias cfg_width: widgetWidth.value |     property alias cfg_width: widgetWidth.value | ||||||
|     property alias cfg_interval: update.value |     property alias cfg_interval: update.value | ||||||
|     property alias cfg_queueLimit: queueLimit.value |     property alias cfg_queueLimit: queueLimit.value | ||||||
|     property string cfg_tempUnits: tempUnits.currentText |     property string cfg_tempUnits: tempUnits.value | ||||||
|     property alias cfg_customTime: customTime.text |     property alias cfg_customTime: customTime.value | ||||||
|     property alias cfg_customUptime: customUptime.text |     property alias cfg_customUptime: customUptime.value | ||||||
|     property alias cfg_acOnline: acOnline.text |     property alias cfg_acOnline: acOnline.value | ||||||
|     property alias cfg_acOffline: acOffline.text |     property alias cfg_acOffline: acOffline.value | ||||||
|  |     property alias cfg_telemetryCount: telemetryCount.value | ||||||
|  |     property alias cfg_telemetryRemote: telemetryRemote.checked | ||||||
|  |     property alias cfg_telemetryId: telemetryId.value | ||||||
|  |  | ||||||
|  |  | ||||||
|     Column { |     Column { | ||||||
|         id: pageColumn |         id: pageColumn | ||||||
|         anchors.fill: parent |         anchors.fill: parent | ||||||
|         Row { |  | ||||||
|             height: implicitHeight |         CheckBoxSelector { | ||||||
|             width: parent.width |  | ||||||
|             QtControls.Label { |  | ||||||
|                 height: parent.heigth |  | ||||||
|                 width: parent.width * 2 / 5 |  | ||||||
|             } |  | ||||||
|             QtControls.CheckBox { |  | ||||||
|             id: background |             id: background | ||||||
|                 width: parent.width * 3 / 5 |  | ||||||
|             text: i18n("Enable background") |             text: i18n("Enable background") | ||||||
|         } |         } | ||||||
|         } |  | ||||||
|  |  | ||||||
|         Row { |         CheckBoxSelector { | ||||||
|             height: implicitHeight |  | ||||||
|             width: parent.width |  | ||||||
|             QtControls.Label { |  | ||||||
|                 height: parent.heigth |  | ||||||
|                 width: parent.width * 2 / 5 |  | ||||||
|             } |  | ||||||
|             QtControls.CheckBox { |  | ||||||
|             id: translate |             id: translate | ||||||
|                 width: parent.width * 3 / 5 |  | ||||||
|             text: i18n("Translate strings") |             text: i18n("Translate strings") | ||||||
|         } |         } | ||||||
|         } |  | ||||||
|  |  | ||||||
|         Row { |         CheckBoxSelector { | ||||||
|             height: implicitHeight |  | ||||||
|             width: parent.width |  | ||||||
|             QtControls.Label { |  | ||||||
|                 height: parent.heigth |  | ||||||
|                 width: parent.width * 2 / 5 |  | ||||||
|             } |  | ||||||
|             QtControls.CheckBox { |  | ||||||
|             id: wrapNewLines |             id: wrapNewLines | ||||||
|                 width: parent.width * 3 / 5 |  | ||||||
|             text: i18n("Wrap new lines") |             text: i18n("Wrap new lines") | ||||||
|         } |         } | ||||||
|         } |  | ||||||
|  |  | ||||||
|         Row { |         CheckBoxSelector { | ||||||
|             height: implicitHeight |  | ||||||
|             width: parent.width |  | ||||||
|             QtControls.Label { |  | ||||||
|                 height: parent.heigth |  | ||||||
|                 width: parent.width * 2 / 5 |  | ||||||
|             } |  | ||||||
|             QtControls.CheckBox { |  | ||||||
|             id: wordWrap |             id: wordWrap | ||||||
|                 width: parent.width * 3 / 5 |  | ||||||
|             text: i18n("Enable word wrap") |             text: i18n("Enable word wrap") | ||||||
|         } |         } | ||||||
|         } |  | ||||||
|  |  | ||||||
|         Row { |         CheckBoxSelector { | ||||||
|             height: implicitHeight |  | ||||||
|             width: parent.width |  | ||||||
|             QtControls.Label { |  | ||||||
|                 height: parent.heigth |  | ||||||
|                 width: parent.width * 2 / 5 |  | ||||||
|             } |  | ||||||
|             QtControls.CheckBox { |  | ||||||
|             id: notify |             id: notify | ||||||
|                 width: parent.width * 3 / 5 |  | ||||||
|             text: i18n("Enable notifications") |             text: i18n("Enable notifications") | ||||||
|         } |         } | ||||||
|         } |  | ||||||
|  |  | ||||||
|         Row { |         CheckBoxSelector { | ||||||
|             height: implicitHeight |  | ||||||
|             width: parent.width |  | ||||||
|             QtControls.Label { |  | ||||||
|                 height: parent.heigth |  | ||||||
|                 width: parent.width * 2 / 5 |  | ||||||
|             } |  | ||||||
|             QtControls.CheckBox { |  | ||||||
|             id: updates |             id: updates | ||||||
|                 width: parent.width * 3 / 5 |  | ||||||
|             text: i18n("Check updates on startup") |             text: i18n("Check updates on startup") | ||||||
|         } |         } | ||||||
|  |  | ||||||
|  |         CheckBoxSelector { | ||||||
|  |             id: optimize | ||||||
|  |             text: i18n("Optimize subscription") | ||||||
|         } |         } | ||||||
|  |  | ||||||
|         Row { |         IntegerSelector { | ||||||
|             height: implicitHeight |  | ||||||
|             width: parent.width |  | ||||||
|             QtControls.Label { |  | ||||||
|                 height: parent.height |  | ||||||
|                 width: parent.width * 2 / 5 |  | ||||||
|                 horizontalAlignment: Text.AlignRight |  | ||||||
|                 verticalAlignment: Text.AlignVCenter |  | ||||||
|                 text: i18n("Widget height, px") |  | ||||||
|             } |  | ||||||
|             QtControls.SpinBox { |  | ||||||
|             id: widgetHeight |             id: widgetHeight | ||||||
|                 width: parent.width * 3 / 5 |  | ||||||
|                 minimumValue: 0 |  | ||||||
|             maximumValue: 4096 |             maximumValue: 4096 | ||||||
|  |             minimumValue: 0 | ||||||
|             stepSize: 50 |             stepSize: 50 | ||||||
|  |             text: i18n("Widget height, px") | ||||||
|             value: plasmoid.configuration.height |             value: plasmoid.configuration.height | ||||||
|         } |         } | ||||||
|         } |  | ||||||
|  |  | ||||||
|         Row { |         IntegerSelector { | ||||||
|             height: implicitHeight |  | ||||||
|             width: parent.width |  | ||||||
|             QtControls.Label { |  | ||||||
|                 height: parent.height |  | ||||||
|                 width: parent.width * 2 / 5 |  | ||||||
|                 horizontalAlignment: Text.AlignRight |  | ||||||
|                 verticalAlignment: Text.AlignVCenter |  | ||||||
|                 text: i18n("Widget width, px") |  | ||||||
|             } |  | ||||||
|             QtControls.SpinBox { |  | ||||||
|             id: widgetWidth |             id: widgetWidth | ||||||
|                 width: parent.width * 3 / 5 |  | ||||||
|                 minimumValue: 0 |  | ||||||
|             maximumValue: 4096 |             maximumValue: 4096 | ||||||
|  |             minimumValue: 0 | ||||||
|             stepSize: 50 |             stepSize: 50 | ||||||
|  |             text: i18n("Widget width, px") | ||||||
|             value: plasmoid.configuration.width |             value: plasmoid.configuration.width | ||||||
|         } |         } | ||||||
|         } |  | ||||||
|  |  | ||||||
|         Row { |         IntegerSelector { | ||||||
|             height: implicitHeight |  | ||||||
|             width: parent.width |  | ||||||
|             QtControls.Label { |  | ||||||
|                 height: parent.height |  | ||||||
|                 width: parent.width * 2 / 5 |  | ||||||
|                 horizontalAlignment: Text.AlignRight |  | ||||||
|                 verticalAlignment: Text.AlignVCenter |  | ||||||
|                 text: i18n("Time interval") |  | ||||||
|             } |  | ||||||
|             QtControls.SpinBox { |  | ||||||
|             id: update |             id: update | ||||||
|                 width: parent.width * 3 / 5 |  | ||||||
|                 minimumValue: 1000 |  | ||||||
|             maximumValue: 10000 |             maximumValue: 10000 | ||||||
|  |             minimumValue: 1000 | ||||||
|             stepSize: 500 |             stepSize: 500 | ||||||
|  |             text: i18n("Time interval") | ||||||
|             value: plasmoid.configuration.interval |             value: plasmoid.configuration.interval | ||||||
|         } |         } | ||||||
|         } |  | ||||||
|  |  | ||||||
|         Row { |         IntegerSelector { | ||||||
|             height: implicitHeight |  | ||||||
|             width: parent.width |  | ||||||
|             QtControls.Label { |  | ||||||
|                 height: parent.height |  | ||||||
|                 width: parent.width * 2 / 5 |  | ||||||
|                 horizontalAlignment: Text.AlignRight |  | ||||||
|                 verticalAlignment: Text.AlignVCenter |  | ||||||
|                 text: i18n("Messages queue limit") |  | ||||||
|             } |  | ||||||
|             QtControls.SpinBox { |  | ||||||
|             id: queueLimit |             id: queueLimit | ||||||
|                 width: parent.width * 3 / 5 |  | ||||||
|                 minimumValue: 0 |  | ||||||
|             maximumValue: 99 |             maximumValue: 99 | ||||||
|  |             minimumValue: 0 | ||||||
|             stepSize: 1 |             stepSize: 1 | ||||||
|  |             text: i18n("Messages queue limit") | ||||||
|             value: plasmoid.configuration.queueLimit |             value: plasmoid.configuration.queueLimit | ||||||
|         } |         } | ||||||
|         } |  | ||||||
|  |  | ||||||
|         Row { |         ComboBoxSelector { | ||||||
|             height: implicitHeight |  | ||||||
|             width: parent.width |  | ||||||
|             QtControls.Label { |  | ||||||
|                 height: parent.height |  | ||||||
|                 width: parent.width * 2 / 5 |  | ||||||
|                 horizontalAlignment: Text.AlignRight |  | ||||||
|                 verticalAlignment: Text.AlignVCenter |  | ||||||
|                 text: i18n("Temperature units") |  | ||||||
|             } |  | ||||||
|             QtControls.ComboBox { |  | ||||||
|             id: tempUnits |             id: tempUnits | ||||||
|                 width: parent.width * 3 / 5 |  | ||||||
|                 textRole: "label" |  | ||||||
|             model: [ |             model: [ | ||||||
|                 { |                 { | ||||||
|                     'label': i18n("Celsius"), |                     'label': i18n("Celsius"), | ||||||
| @ -266,130 +163,93 @@ Item { | |||||||
|                     'name': "kcal/mol" |                     'name': "kcal/mol" | ||||||
|                 } |                 } | ||||||
|             ] |             ] | ||||||
|                 onCurrentIndexChanged: cfg_tempUnits = model[currentIndex]["name"] |             text: i18n("Temperature units") | ||||||
|                 Component.onCompleted: { |             value: plasmoid.configuration.tempUnits | ||||||
|                     if (debug) console.debug() |             onValueEdited: cfg_tempUnits = newValue | ||||||
|                     for (var i = 0; i < model.length; i++) { |  | ||||||
|                         if (model[i]["name"] == plasmoid.configuration.tempUnits) { |  | ||||||
|                             if (debug) console.info("Found", model[i]["name"], "on", i) |  | ||||||
|                             tempUnits.currentIndex = i; |  | ||||||
|                         } |  | ||||||
|                     } |  | ||||||
|                 } |  | ||||||
|             } |  | ||||||
|         } |         } | ||||||
|  |  | ||||||
|         Row { |         LineSelector { | ||||||
|             height: implicitHeight |  | ||||||
|             width: parent.width |  | ||||||
|             QtControls.Label { |  | ||||||
|                 height: parent.height |  | ||||||
|                 width: parent.width * 2 / 5 |  | ||||||
|                 horizontalAlignment: Text.AlignRight |  | ||||||
|                 verticalAlignment: Text.AlignVCenter |  | ||||||
|                 text: i18n("Custom time format") |  | ||||||
|             } |  | ||||||
|             QtControls.TextField { |  | ||||||
|             id: customTime |             id: customTime | ||||||
|                 width: parent.width * 3 / 5 |             text: i18n("Custom time format") | ||||||
|                 text: plasmoid.configuration.customTime |             value: plasmoid.configuration.customTime | ||||||
|             } |  | ||||||
|         } |         } | ||||||
|  |  | ||||||
|         Row { |         LineSelector { | ||||||
|             height: implicitHeight |  | ||||||
|             width: parent.width |  | ||||||
|             QtControls.Label { |  | ||||||
|                 height: parent.height |  | ||||||
|                 width: parent.width * 2 / 5 |  | ||||||
|                 horizontalAlignment: Text.AlignRight |  | ||||||
|                 verticalAlignment: Text.AlignVCenter |  | ||||||
|                 text: i18n("Custom uptime format") |  | ||||||
|             } |  | ||||||
|             QtControls.TextField { |  | ||||||
|             id: customUptime |             id: customUptime | ||||||
|                 width: parent.width * 3 / 5 |             text: i18n("Custom uptime format") | ||||||
|                 text: plasmoid.configuration.customUptime |             value: plasmoid.configuration.customUptime | ||||||
|             } |  | ||||||
|         } |         } | ||||||
|  |  | ||||||
|         Row { |         LineSelector { | ||||||
|             height: implicitHeight |  | ||||||
|             width: parent.width |  | ||||||
|             QtControls.Label { |  | ||||||
|                 height: parent.height |  | ||||||
|                 width: parent.width * 2 / 5 |  | ||||||
|                 horizontalAlignment: Text.AlignRight |  | ||||||
|                 verticalAlignment: Text.AlignVCenter |  | ||||||
|                 text: i18n("AC online tag") |  | ||||||
|             } |  | ||||||
|             QtControls.TextField { |  | ||||||
|             id: acOnline |             id: acOnline | ||||||
|                 width: parent.width * 3 / 5 |             text: i18n("AC online tag") | ||||||
|                 text: plasmoid.configuration.acOnline |             value: plasmoid.configuration.acOnline | ||||||
|             } |  | ||||||
|         } |         } | ||||||
|  |  | ||||||
|         Row { |         LineSelector { | ||||||
|             height: implicitHeight |  | ||||||
|             width: parent.width |  | ||||||
|             QtControls.Label { |  | ||||||
|                 height: parent.height |  | ||||||
|                 width: parent.width * 2 / 5 |  | ||||||
|                 horizontalAlignment: Text.AlignRight |  | ||||||
|                 verticalAlignment: Text.AlignVCenter |  | ||||||
|                 text: i18n("AC offline tag") |  | ||||||
|             } |  | ||||||
|             QtControls.TextField { |  | ||||||
|             id: acOffline |             id: acOffline | ||||||
|                 width: parent.width * 3 / 5 |             text: i18n("AC offline tag") | ||||||
|                 text: plasmoid.configuration.acOffline |             value: plasmoid.configuration.acOffline | ||||||
|  |         } | ||||||
|  |  | ||||||
|  |         QtControls.GroupBox { | ||||||
|  |             height: implicitHeight | ||||||
|  |             width: parent.width | ||||||
|  |             title: i18n("Actions") | ||||||
|  |  | ||||||
|  |             Column { | ||||||
|  |                 height: implicitHeight | ||||||
|  |                 width: parent.width | ||||||
|  |                 ButtonSelector { | ||||||
|  |                     value: i18n("Drop key cache") | ||||||
|  |                     onButtonActivated: awActions.dropCache() | ||||||
|  |                 } | ||||||
|  |                 ButtonSelector { | ||||||
|  |                     ExportDialog { | ||||||
|  |                         id: saveConfigAs | ||||||
|  |                         configuration: plasmoid.configuration | ||||||
|  |                     } | ||||||
|  |                     value: i18n("Export configuration") | ||||||
|  |                     onButtonActivated: saveConfigAs.open() | ||||||
|  |                 } | ||||||
|  |                 ButtonSelector { | ||||||
|  |                     ImportDialog { | ||||||
|  |                         id: loadConfigFrom | ||||||
|  |                         onConfigurationReceived: { | ||||||
|  |                             for (var key in configuration) | ||||||
|  |                                 plasmoid.configuration[key] = configuration[key] | ||||||
|  |                         } | ||||||
|  |                     } | ||||||
|  |                     value: i18n("Import configuration") | ||||||
|  |                     onButtonActivated: loadConfigFrom.open() | ||||||
|  |                 } | ||||||
|             } |             } | ||||||
|         } |         } | ||||||
|  |  | ||||||
|         Row { |         QtControls.GroupBox { | ||||||
|             height: implicitHeight |             height: implicitHeight | ||||||
|             width: parent.width |             width: parent.width | ||||||
|             QtControls.Label { |             title: i18n("Telemetry") | ||||||
|                 height: parent.height |  | ||||||
|                 width: parent.width * 2 / 5 |  | ||||||
|             } |  | ||||||
|             QtControls.Button { |  | ||||||
|                 width: parent.width * 3 / 5 |  | ||||||
|                 text: i18n("Drop key cache") |  | ||||||
|                 onClicked: awActions.dropCache() |  | ||||||
|             } |  | ||||||
|         } |  | ||||||
|  |  | ||||||
|         Row { |             Column { | ||||||
|                 height: implicitHeight |                 height: implicitHeight | ||||||
|                 width: parent.width |                 width: parent.width | ||||||
|             QtControls.Label { |                 CheckBoxSelector { | ||||||
|                 height: parent.height |                     id: telemetryRemote | ||||||
|                 width: parent.width * 2 / 5 |                     text: i18n("Enable remote telemetry") | ||||||
|                 } |                 } | ||||||
|             QtControls.Button { |                 IntegerSelector { | ||||||
|                 width: parent.width * 3 / 5 |                     id: telemetryCount | ||||||
|                 text: i18n("Export configuration") |                     maximumValue: 10000 | ||||||
|                 onClicked: awConfig.exportConfiguration(plasmoid.configuration) |                     minimumValue: 0 | ||||||
|  |                     stepSize: 50 | ||||||
|  |                     text: i18n("History count") | ||||||
|  |                     value: plasmoid.configuration.telemetryCount | ||||||
|                 } |                 } | ||||||
|         } |                 LineSelector { | ||||||
|  |                     id: telemetryId | ||||||
|         Row { |                     text: i18n("Telemetry ID") | ||||||
|             height: implicitHeight |                     value: plasmoid.configuration.telemetryId | ||||||
|             width: parent.width |  | ||||||
|             QtControls.Label { |  | ||||||
|                 height: parent.height |  | ||||||
|                 width: parent.width * 2 / 5 |  | ||||||
|             } |  | ||||||
|             QtControls.Button { |  | ||||||
|                 width: parent.width * 3 / 5 |  | ||||||
|                 text: i18n("Import configuration") |  | ||||||
|                 onClicked: { |  | ||||||
|                     if (debug) console.debug() |  | ||||||
|                     var importConfig = awConfig.importConfiguration() |  | ||||||
|                     for (var key in importConfig) |  | ||||||
|                         plasmoid.configuration[key] = importConfig[key] |  | ||||||
|                 } |                 } | ||||||
|             } |             } | ||||||
|         } |         } | ||||||
|  | |||||||
| @ -16,11 +16,9 @@ | |||||||
|  ***************************************************************************/ |  ***************************************************************************/ | ||||||
|  |  | ||||||
| import QtQuick 2.0 | import QtQuick 2.0 | ||||||
| import QtQuick.Controls 1.3 as QtControls |  | ||||||
| import QtQuick.Controls.Styles 1.3 as QtStyles |  | ||||||
| import QtQuick.Dialogs 1.1 as QtDialogs |  | ||||||
|  |  | ||||||
| import org.kde.plasma.private.awesomewidget 1.0 | import org.kde.plasma.private.awesomewidget 1.0 | ||||||
|  | import "." | ||||||
|  |  | ||||||
|  |  | ||||||
| Item { | Item { | ||||||
| @ -44,195 +42,68 @@ Item { | |||||||
|         87: 5 |         87: 5 | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     property alias cfg_fontFamily: selectFont.text |     property alias cfg_fontFamily: font.value | ||||||
|     property alias cfg_fontSize: fontSize.value |     property alias cfg_fontSize: fontSize.value | ||||||
|     property string cfg_fontWeight: fontWeight.currentText |     property string cfg_fontWeight: fontWeight.value | ||||||
|     property string cfg_fontStyle: fontStyle.currentText |     property string cfg_fontStyle: fontStyle.value | ||||||
|     property alias cfg_fontColor: selectColor.text |     property alias cfg_fontColor: selectColor.value | ||||||
|  |     property alias cfg_textStyleColor: selectStyleColor.value | ||||||
|  |     property string cfg_textStyle: textStyle.value | ||||||
|  |  | ||||||
|  |  | ||||||
|     Column { |     Column { | ||||||
|         id: pageColumn |         id: pageColumn | ||||||
|         anchors.fill: parent |         anchors.fill: parent | ||||||
|         Row { |  | ||||||
|             height: implicitHeight |         FontSelector { | ||||||
|             width: parent.width |             id: font | ||||||
|             QtControls.Label { |  | ||||||
|                 height: parent.height |  | ||||||
|                 width: parent.width / 3 |  | ||||||
|                 horizontalAlignment: Text.AlignRight |  | ||||||
|                 verticalAlignment: Text.AlignVCenter |  | ||||||
|             text: i18n("Font") |             text: i18n("Font") | ||||||
|             } |             value: plasmoid.configuration.fontFamily | ||||||
|             QtControls.Button { |  | ||||||
|                 id: selectFont |  | ||||||
|                 width: parent.width * 2 / 3 |  | ||||||
|                 text: plasmoid.configuration.fontFamily |  | ||||||
|                 onClicked: { |  | ||||||
|                     if (debug) console.debug() |  | ||||||
|                     fontDialog.setFont() |  | ||||||
|                     fontDialog.visible = true |  | ||||||
|                 } |  | ||||||
|             } |  | ||||||
|         } |         } | ||||||
|  |  | ||||||
|         Row { |         IntegerSelector { | ||||||
|             height: implicitHeight |  | ||||||
|             width: parent.width |  | ||||||
|             QtControls.Label { |  | ||||||
|                 height: parent.height |  | ||||||
|                 width: parent.width / 3 |  | ||||||
|                 horizontalAlignment: Text.AlignRight |  | ||||||
|                 verticalAlignment: Text.AlignVCenter |  | ||||||
|                 text: i18n("Font size") |  | ||||||
|             } |  | ||||||
|             QtControls.SpinBox { |  | ||||||
|             id: fontSize |             id: fontSize | ||||||
|                 width: parent.width * 2 / 3 |  | ||||||
|                 minimumValue: 8 |  | ||||||
|             maximumValue: 32 |             maximumValue: 32 | ||||||
|  |             minimumValue: 8 | ||||||
|             stepSize: 1 |             stepSize: 1 | ||||||
|  |             text: i18n("Font size") | ||||||
|             value: plasmoid.configuration.fontSize |             value: plasmoid.configuration.fontSize | ||||||
|         } |         } | ||||||
|         } |  | ||||||
|  |  | ||||||
|         Row { |         ComboBoxSelector { | ||||||
|             height: implicitHeight |  | ||||||
|             width: parent.width |  | ||||||
|             QtControls.Label { |  | ||||||
|                 height: parent.height |  | ||||||
|                 width: parent.width / 3 |  | ||||||
|                 horizontalAlignment: Text.AlignRight |  | ||||||
|                 verticalAlignment: Text.AlignVCenter |  | ||||||
|                 text: i18n("Font weight") |  | ||||||
|             } |  | ||||||
|             QtControls.ComboBox { |  | ||||||
|             id: fontWeight |             id: fontWeight | ||||||
|                 width: parent.width * 2 / 3 |             model: General.fontWeightModel | ||||||
|                 textRole: "label" |             text: i18n("Font weight") | ||||||
|                 model: [ |             value: plasmoid.configuration.fontWeight | ||||||
|                     { |             onValueEdited: cfg_fontWeight = newValue | ||||||
|                         'label': i18n("light"), |  | ||||||
|                         'name': "light" |  | ||||||
|                     }, |  | ||||||
|                     { |  | ||||||
|                         'label': i18n("normal"), |  | ||||||
|                         'name': "normal" |  | ||||||
|                     }, |  | ||||||
|                     { |  | ||||||
|                         'label': i18n("demi bold"), |  | ||||||
|                         'name': "demibold" |  | ||||||
|                     }, |  | ||||||
|                     { |  | ||||||
|                         'label': i18n("bold"), |  | ||||||
|                         'name': "bold" |  | ||||||
|                     }, |  | ||||||
|                     { |  | ||||||
|                         'label': i18n("black"), |  | ||||||
|                         'name': "black" |  | ||||||
|                     } |  | ||||||
|                 ] |  | ||||||
|                 onCurrentIndexChanged: cfg_fontWeight = model[currentIndex]["name"] |  | ||||||
|                 Component.onCompleted: { |  | ||||||
|                     if (debug) console.debug() |  | ||||||
|                     for (var i = 0; i < model.length; i++) { |  | ||||||
|                         if (model[i]["name"] == plasmoid.configuration.fontWeight) { |  | ||||||
|                             if (debug) console.info("Found", model[i]["name"], "on", i) |  | ||||||
|                             fontWeight.currentIndex = i; |  | ||||||
|                         } |  | ||||||
|                     } |  | ||||||
|                 } |  | ||||||
|             } |  | ||||||
|         } |         } | ||||||
|  |  | ||||||
|         Row { |         ComboBoxSelector { | ||||||
|             height: implicitHeight |  | ||||||
|             width: parent.width |  | ||||||
|             QtControls.Label { |  | ||||||
|                 height: parent.height |  | ||||||
|                 width: parent.width / 3 |  | ||||||
|                 horizontalAlignment: Text.AlignRight |  | ||||||
|                 verticalAlignment: Text.AlignVCenter |  | ||||||
|                 text: i18n("Font style") |  | ||||||
|             } |  | ||||||
|             QtControls.ComboBox { |  | ||||||
|             id: fontStyle |             id: fontStyle | ||||||
|                 width: parent.width * 2 / 3 |             model: General.fontStyleModel | ||||||
|                 textRole: "label" |             text: i18n("Font style") | ||||||
|                 model: [ |             value: plasmoid.configuration.fontStyle | ||||||
|                     { |             onValueEdited: cfg_fontStyle = newValue | ||||||
|                         'label': i18n("normal"), |  | ||||||
|                         'name': "normal" |  | ||||||
|                     }, |  | ||||||
|                     { |  | ||||||
|                         'label': i18n("italic"), |  | ||||||
|                         'name': "italic" |  | ||||||
|                     } |  | ||||||
|                 ] |  | ||||||
|                 onCurrentIndexChanged: cfg_fontStyle = model[currentIndex]["name"] |  | ||||||
|                 Component.onCompleted: { |  | ||||||
|                     if (debug) console.debug() |  | ||||||
|                     for (var i = 0; i < model.length; i++) { |  | ||||||
|                         if (model[i]["name"] == plasmoid.configuration.fontStyle) { |  | ||||||
|                             if (debug) console.info("Found", model[i]["name"], "on", i) |  | ||||||
|                             fontStyle.currentIndex = i; |  | ||||||
|                         } |  | ||||||
|                     } |  | ||||||
|                 } |  | ||||||
|             } |  | ||||||
|         } |         } | ||||||
|  |  | ||||||
|         Row { |         ColorSelector { | ||||||
|             height: implicitHeight |  | ||||||
|             width: parent.width |  | ||||||
|             QtControls.Label { |  | ||||||
|                 height: parent.height |  | ||||||
|                 width: parent.width / 3 |  | ||||||
|                 horizontalAlignment: Text.AlignRight |  | ||||||
|                 verticalAlignment: Text.AlignVCenter |  | ||||||
|                 text: i18n("Font color") |  | ||||||
|             } |  | ||||||
|             QtControls.Button { |  | ||||||
|             id: selectColor |             id: selectColor | ||||||
|                 width: parent.width * 2 / 3 |             text: i18n("Font color") | ||||||
|                 style: QtStyles.ButtonStyle { |             value: plasmoid.configuration.fontColor | ||||||
|                     background: Rectangle { |  | ||||||
|                         color: plasmoid.configuration.fontColor |  | ||||||
|                     } |  | ||||||
|                 } |  | ||||||
|                 text: plasmoid.configuration.fontColor |  | ||||||
|                 onClicked: colorDialog.visible = true |  | ||||||
|             } |  | ||||||
|         } |  | ||||||
|         } |         } | ||||||
|  |  | ||||||
|     QtDialogs.ColorDialog { |         ComboBoxSelector { | ||||||
|         id: colorDialog |             id: textStyle | ||||||
|         title: i18n("Select a color") |             model: General.textStyleModel | ||||||
|         color: selectColor.text |             text: i18n("Style") | ||||||
|         onAccepted: selectColor.text = colorDialog.color |             value: plasmoid.configuration.textStyle | ||||||
|  |             onValueEdited: cfg_textStyle = newValue | ||||||
|         } |         } | ||||||
|  |  | ||||||
|     QtDialogs.FontDialog { |         ColorSelector { | ||||||
|         id: fontDialog |             id: selectStyleColor | ||||||
|         title: i18n("Select a font") |             text: i18n("Style color") | ||||||
|         signal setFont |             value: plasmoid.configuration.textStyleColor | ||||||
|  |  | ||||||
|         onAccepted: { |  | ||||||
|             if (debug) console.debug() |  | ||||||
|             selectFont.text = fontDialog.font.family |  | ||||||
|             fontSize.value = fontDialog.font.pointSize |  | ||||||
|             fontStyle.currentIndex = fontDialog.font.italic ? 1 : 0 |  | ||||||
|             fontWeight.currentIndex = weight[fontDialog.font.weight] |  | ||||||
|         } |  | ||||||
|         onSetFont: { |  | ||||||
|             if (debug) console.debug() |  | ||||||
|             fontDialog.font = Qt.font({ |  | ||||||
|                 family: selectFont.text, |  | ||||||
|                 pointSize: fontSize.value > 0 ? fontSize.value : 12, |  | ||||||
|                 italic: fontStyle.currentIndex == 1, |  | ||||||
|                 weight: Font.Normal, |  | ||||||
|             }) |  | ||||||
|         } |         } | ||||||
|     } |     } | ||||||
|  |  | ||||||
|  | |||||||
| @ -52,21 +52,10 @@ Item { | |||||||
|             height: implicitHeight |             height: implicitHeight | ||||||
|             width: parent.width |             width: parent.width | ||||||
|             title: i18n("ACPI") |             title: i18n("ACPI") | ||||||
|             Row { |             LineSelector { | ||||||
|                 height: implicitHeight |  | ||||||
|                 width: parent.width |  | ||||||
|                 QtControls.Label { |  | ||||||
|                     height: parent.height |  | ||||||
|                     width: parent.width * 2 / 5 |  | ||||||
|                     horizontalAlignment: Text.AlignRight |  | ||||||
|                     verticalAlignment: Text.AlignVCenter |  | ||||||
|                 text: i18n("ACPI path") |                 text: i18n("ACPI path") | ||||||
|                 } |                 value: cfg_dataengine["ACPIPATH"] | ||||||
|                 QtControls.TextField { |                 onValueEdited: cfg_dataengine["ACPIPATH"] = newValue | ||||||
|                     width: parent.width * 3 / 5 |  | ||||||
|                     text: cfg_dataengine["ACPIPATH"] |  | ||||||
|                     onEditingFinished: cfg_dataengine["ACPIPATH"] = text |  | ||||||
|                 } |  | ||||||
|             } |             } | ||||||
|         } |         } | ||||||
|  |  | ||||||
| @ -74,30 +63,28 @@ Item { | |||||||
|             height: implicitHeight |             height: implicitHeight | ||||||
|             width: parent.width |             width: parent.width | ||||||
|             title: i18n("GPU") |             title: i18n("GPU") | ||||||
|             Row { |             ComboBoxSelector { | ||||||
|                 height: implicitHeight |                 model: [ | ||||||
|                 width: parent.width |                     { | ||||||
|                 QtControls.Label { |                         'label': "auto", | ||||||
|                     height: parent.height |                         'name': "auto" | ||||||
|                     width: parent.width * 2 / 5 |                     }, | ||||||
|                     horizontalAlignment: Text.AlignRight |                     { | ||||||
|                     verticalAlignment: Text.AlignVCenter |                         'label': "disable", | ||||||
|  |                         'name': "disable" | ||||||
|  |                     }, | ||||||
|  |                     { | ||||||
|  |                         'label': "ati", | ||||||
|  |                         'name': "ati" | ||||||
|  |                     }, | ||||||
|  |                     { | ||||||
|  |                         'label': "nvidia", | ||||||
|  |                         'name': "nvidia" | ||||||
|  |                     } | ||||||
|  |                 ] | ||||||
|                 text: i18n("GPU device") |                 text: i18n("GPU device") | ||||||
|                 } |                 value: cfg_dataengine["GPUDEV"] | ||||||
|                 QtControls.ComboBox { |                 onValueEdited: cfg_dataengine["GPUDEV"] = newValue | ||||||
|                     id: gpuDev |  | ||||||
|                     width: parent.width * 3 / 5 |  | ||||||
|                     model: ["auto", "disable", "ati", "nvidia"] |  | ||||||
|                     Component.onCompleted: { |  | ||||||
|                         if (debug) console.debug() |  | ||||||
|                         for (var i=0; i<model.length; i++) { |  | ||||||
|                             if (model[i] == cfg_dataengine["GPUDEV"]) { |  | ||||||
|                                 if (debug) console.info("Found", model[i], "on", i) |  | ||||||
|                                 currentIndex = i; |  | ||||||
|                             } |  | ||||||
|                         } |  | ||||||
|                     } |  | ||||||
|                 } |  | ||||||
|             } |             } | ||||||
|         } |         } | ||||||
|  |  | ||||||
| @ -108,37 +95,17 @@ Item { | |||||||
|             Column { |             Column { | ||||||
|                 height: implicitHeight |                 height: implicitHeight | ||||||
|                 width: parent.width |                 width: parent.width | ||||||
|                 Row { |                 ComboBoxSelector { | ||||||
|                     height: implicitHeight |  | ||||||
|                     width: parent.width |  | ||||||
|                     QtControls.Label { |  | ||||||
|                         height: parent.height |  | ||||||
|                         width: parent.width * 2 / 5 |  | ||||||
|                         horizontalAlignment: Text.AlignRight |  | ||||||
|                         verticalAlignment: Text.AlignVCenter |  | ||||||
|                         text: i18n("HDD") |  | ||||||
|                     } |  | ||||||
|                     QtControls.ComboBox { |  | ||||||
|                     id: hdd |                     id: hdd | ||||||
|                         width: parent.width * 3 / 5 |                     text: i18n("HDD") | ||||||
|                     } |                     value: cfg_dataengine["HDDDEV"] | ||||||
|  |                     onValueEdited: cfg_dataengine["HDDDEV"] = newValue | ||||||
|                 } |                 } | ||||||
|  |  | ||||||
|                 Row { |                 LineSelector { | ||||||
|                     height: implicitHeight |  | ||||||
|                     width: parent.width |  | ||||||
|                     QtControls.Label { |  | ||||||
|                         height: parent.height |  | ||||||
|                         width: parent.width * 2 / 5 |  | ||||||
|                         horizontalAlignment: Text.AlignRight |  | ||||||
|                         verticalAlignment: Text.AlignVCenter |  | ||||||
|                     text: i18n("hddtemp cmd") |                     text: i18n("hddtemp cmd") | ||||||
|                     } |                     value: cfg_dataengine["HDDTEMPCMD"] | ||||||
|                     QtControls.TextField { |                     onValueEdited: cfg_dataengine["HDDTEMPCMD"] = newValue | ||||||
|                         width: parent.width * 3 / 5 |  | ||||||
|                         text: cfg_dataengine["HDDTEMPCMD"] |  | ||||||
|                         onEditingFinished: cfg_dataengine["HDDTEMPCMD"] = text |  | ||||||
|                     } |  | ||||||
|                 } |                 } | ||||||
|             } |             } | ||||||
|         } |         } | ||||||
| @ -150,107 +117,93 @@ Item { | |||||||
|             Column { |             Column { | ||||||
|                 height: implicitHeight |                 height: implicitHeight | ||||||
|                 width: parent.width |                 width: parent.width | ||||||
|                 Row { |                 IntegerSelector { | ||||||
|                     height: implicitHeight |  | ||||||
|                     width: parent.width |  | ||||||
|                     QtControls.Label { |  | ||||||
|                         height: parent.height |  | ||||||
|                         width: parent.width * 2 / 5 |  | ||||||
|                         horizontalAlignment: Text.AlignRight |  | ||||||
|                         verticalAlignment: Text.AlignVCenter |  | ||||||
|                         text: i18n("Player data symbols") |  | ||||||
|                     } |  | ||||||
|                     QtControls.SpinBox { |  | ||||||
|                         width: parent.width * 3 / 5 |  | ||||||
|                         minimumValue: 1 |  | ||||||
|                     maximumValue: 100 |                     maximumValue: 100 | ||||||
|  |                     minimumValue: 1 | ||||||
|                     stepSize: 1 |                     stepSize: 1 | ||||||
|  |                     text: i18n("Player data symbols") | ||||||
|                     value: cfg_dataengine["PLAYERSYMBOLS"] |                     value: cfg_dataengine["PLAYERSYMBOLS"] | ||||||
|                         onEditingFinished: cfg_dataengine["PLAYERSYMBOLS"] = value |                     onValueEdited: cfg_dataengine["PLAYERSYMBOLS"] = newValue | ||||||
|                     } |  | ||||||
|                 } |                 } | ||||||
|  |  | ||||||
|                 Row { |                 ComboBoxSelector { | ||||||
|                     height: implicitHeight |                     model: [ | ||||||
|                     width: parent.width |                         { | ||||||
|                     QtControls.Label { |                             'label': "disable", | ||||||
|                         height: parent.height |                             'name': "disable" | ||||||
|                         width: parent.width * 2 / 5 |                         }, | ||||||
|                         horizontalAlignment: Text.AlignRight |                         { | ||||||
|                         verticalAlignment: Text.AlignVCenter |                             'label': "mpris", | ||||||
|  |                             'name': "mpris" | ||||||
|  |                         }, | ||||||
|  |                         { | ||||||
|  |                             'label': "mpd", | ||||||
|  |                             'name': "mpd" | ||||||
|  |                         } | ||||||
|  |                     ] | ||||||
|                     text: i18n("Music player") |                     text: i18n("Music player") | ||||||
|                     } |                     value: cfg_dataengine["PLAYER"] | ||||||
|                     QtControls.ComboBox { |                     onValueEdited: cfg_dataengine["PLAYER"] = newValue | ||||||
|                         id: player |  | ||||||
|                         width: parent.width * 3 / 5 |  | ||||||
|                         model: ["disable", "mpris", "mpd"] |  | ||||||
|                         Component.onCompleted: { |  | ||||||
|                             if (debug) console.debug() |  | ||||||
|                             for (var i=0; i<model.length; i++) { |  | ||||||
|                                 if (model[i] == cfg_dataengine["PLAYER"]) { |  | ||||||
|                                     if (debug) console.info("Found", model[i], "on", i) |  | ||||||
|                                     player.currentIndex = i; |  | ||||||
|                                 } |  | ||||||
|                             } |  | ||||||
|                         } |  | ||||||
|                     } |  | ||||||
|                 } |                 } | ||||||
|  |  | ||||||
|                 Row { |                 ComboBoxSelector { | ||||||
|                     height: implicitHeight |  | ||||||
|                     width: parent.width |  | ||||||
|                     QtControls.Label { |  | ||||||
|                         height: parent.height |  | ||||||
|                         width: parent.width * 2 / 5 |  | ||||||
|                         horizontalAlignment: Text.AlignRight |  | ||||||
|                         verticalAlignment: Text.AlignVCenter |  | ||||||
|                         text: i18n("MPRIS player name") |  | ||||||
|                     } |  | ||||||
|                     QtControls.ComboBox { |  | ||||||
|                     id: mpris |                     id: mpris | ||||||
|                         width: parent.width * 3 / 5 |  | ||||||
|                     editable: true |                     editable: true | ||||||
|                         model: ["auto", "amarok", "audacious", "clementine", "deadbeef", |                     model: [ | ||||||
|                                 "vlc", "qmmp", "xmms2", cfg_dataengine["MPRIS"]] |                         { | ||||||
|  |                             'label': 'auto', | ||||||
|  |                             'name': 'auto' | ||||||
|  |                         }, | ||||||
|  |                         { | ||||||
|  |                             'label': 'amarok', | ||||||
|  |                             'name': 'amarok' | ||||||
|  |                         }, | ||||||
|  |                         { | ||||||
|  |                             'label': 'audacious', | ||||||
|  |                             'name': 'audacious' | ||||||
|  |                         }, | ||||||
|  |                         { | ||||||
|  |                             'label': 'clementine', | ||||||
|  |                             'name': 'clementine' | ||||||
|  |                         }, | ||||||
|  |                         { | ||||||
|  |                             'label': 'DeaDBeeF', | ||||||
|  |                             'name': 'DeaDBeeF' | ||||||
|  |                         }, | ||||||
|  |                         { | ||||||
|  |                             'label': 'vlc', | ||||||
|  |                             'name': 'vlc' | ||||||
|  |                         }, | ||||||
|  |                         { | ||||||
|  |                             'label': 'qmmp', | ||||||
|  |                             'name': 'qmmp' | ||||||
|  |                         }, | ||||||
|  |                         { | ||||||
|  |                             'label': 'xmms2', | ||||||
|  |                             'name': 'xmms2' | ||||||
|  |                         }, | ||||||
|  |                         { | ||||||
|  |                             'label': cfg_dataengine["MPRIS"], | ||||||
|  |                             'name': cfg_dataengine["MPRIS"] | ||||||
|  |                         } | ||||||
|  |                     ] | ||||||
|  |                     text: i18n("MPRIS player name") | ||||||
|                     currentIndex: model.length - 1 |                     currentIndex: model.length - 1 | ||||||
|                 } |                 } | ||||||
|                 } |  | ||||||
|  |  | ||||||
|                 Row { |                 LineSelector { | ||||||
|                     height: implicitHeight |  | ||||||
|                     width: parent.width |  | ||||||
|                     QtControls.Label { |  | ||||||
|                         height: parent.height |  | ||||||
|                         width: parent.width * 2 / 5 |  | ||||||
|                         horizontalAlignment: Text.AlignRight |  | ||||||
|                         verticalAlignment: Text.AlignVCenter |  | ||||||
|                     text: i18n("MPD address") |                     text: i18n("MPD address") | ||||||
|                     } |                     value: cfg_dataengine["MPDADDRESS"] | ||||||
|                     QtControls.TextField { |                     onValueEdited: cfg_dataengine["MPDADDRESS"] = newValue | ||||||
|                         width: parent.width * 3 / 5 |  | ||||||
|                         text: cfg_dataengine["MPDADDRESS"] |  | ||||||
|                         onEditingFinished: cfg_dataengine["MPDADDRESS"] = text |  | ||||||
|                     } |  | ||||||
|                 } |                 } | ||||||
|  |  | ||||||
|                 Row { |                 IntegerSelector { | ||||||
|                     height: implicitHeight |  | ||||||
|                     width: parent.width |  | ||||||
|                     QtControls.Label { |  | ||||||
|                         height: parent.height |  | ||||||
|                         width: parent.width * 2 / 5 |  | ||||||
|                         horizontalAlignment: Text.AlignRight |  | ||||||
|                         verticalAlignment: Text.AlignVCenter |  | ||||||
|                         text: i18n("MPD port") |  | ||||||
|                     } |  | ||||||
|                     QtControls.SpinBox { |  | ||||||
|                         width: parent.width * 3 / 5 |  | ||||||
|                         minimumValue: 1000 |  | ||||||
|                     maximumValue: 65535 |                     maximumValue: 65535 | ||||||
|  |                     minimumValue: 1000 | ||||||
|                     stepSize: 1 |                     stepSize: 1 | ||||||
|  |                     text: i18n("MPD port") | ||||||
|                     value: cfg_dataengine["MPDPORT"] |                     value: cfg_dataengine["MPDPORT"] | ||||||
|                         onEditingFinished: cfg_dataengine["MPDPORT"] = value |                     onValueEdited: cfg_dataengine["MPDPORT"] = newValue | ||||||
|                     } |  | ||||||
|                 } |                 } | ||||||
|             } |             } | ||||||
|         } |         } | ||||||
| @ -262,72 +215,29 @@ Item { | |||||||
|             Column { |             Column { | ||||||
|                 height: implicitHeight |                 height: implicitHeight | ||||||
|                 width: parent.width |                 width: parent.width | ||||||
|                 Row { |                 ButtonSelector { | ||||||
|                     height: implicitHeight |                     value: i18n("Custom scripts") | ||||||
|                     width: parent.width |                     onButtonActivated: awKeys.editItem("extscript") | ||||||
|                     QtControls.Label { |  | ||||||
|                         height: parent.height |  | ||||||
|                         width: parent.width * 2 / 5 |  | ||||||
|                         horizontalAlignment: Text.AlignRight |  | ||||||
|                         verticalAlignment: Text.AlignVCenter |  | ||||||
|                         text: i18n("Custom scripts") |  | ||||||
|                     } |  | ||||||
|                     QtControls.Button { |  | ||||||
|                         width: parent.width * 3 / 5 |  | ||||||
|                         text: i18n("Edit scripts") |  | ||||||
|                         onClicked: awKeys.editItem("extscript") |  | ||||||
|                     } |  | ||||||
|                 } |                 } | ||||||
|  |  | ||||||
|                 Row { |                 ButtonSelector { | ||||||
|                     height: implicitHeight |                     value: i18n("Network requests") | ||||||
|                     width: parent.width |                     onButtonActivated: awKeys.editItem("extnetworkrequest") | ||||||
|                     QtControls.Label { |  | ||||||
|                         height: parent.height |  | ||||||
|                         width: parent.width * 2 / 5 |  | ||||||
|                         horizontalAlignment: Text.AlignRight |  | ||||||
|                         verticalAlignment: Text.AlignVCenter |  | ||||||
|                         text: i18n("Quotes monitor") |  | ||||||
|                     } |  | ||||||
|                     QtControls.Button { |  | ||||||
|                         width: parent.width * 3 / 5 |  | ||||||
|                         text: i18n("Edit tickers") |  | ||||||
|                         onClicked: awKeys.editItem("extquotes") |  | ||||||
|                     } |  | ||||||
|                 } |                 } | ||||||
|  |  | ||||||
|                 Row { |                 ButtonSelector { | ||||||
|                     height: implicitHeight |                     value: i18n("Package manager") | ||||||
|                     width: parent.width |                     onButtonActivated: awKeys.editItem("extupgrade") | ||||||
|                     QtControls.Label { |  | ||||||
|                         height: parent.height |  | ||||||
|                         width: parent.width * 2 / 5 |  | ||||||
|                         horizontalAlignment: Text.AlignRight |  | ||||||
|                         verticalAlignment: Text.AlignVCenter |  | ||||||
|                         text: i18n("Package manager") |  | ||||||
|                     } |  | ||||||
|                     QtControls.Button { |  | ||||||
|                         width: parent.width * 3 / 5 |  | ||||||
|                         text: i18n("Edit command") |  | ||||||
|                         onClicked: awKeys.editItem("extupgrade") |  | ||||||
|                     } |  | ||||||
|                 } |                 } | ||||||
|  |  | ||||||
|                 Row { |                 ButtonSelector { | ||||||
|                     height: implicitHeight |                     value: i18n("Quotes monitor") | ||||||
|                     width: parent.width |                     onButtonActivated: awKeys.editItem("extquotes") | ||||||
|                     QtControls.Label { |  | ||||||
|                         height: parent.height |  | ||||||
|                         width: parent.width * 2 / 5 |  | ||||||
|                         horizontalAlignment: Text.AlignRight |  | ||||||
|                         verticalAlignment: Text.AlignVCenter |  | ||||||
|                         text: i18n("Weather") |  | ||||||
|                     } |  | ||||||
|                     QtControls.Button { |  | ||||||
|                         width: parent.width * 3 / 5 |  | ||||||
|                         text: i18n("Edit weather") |  | ||||||
|                         onClicked: awKeys.editItem("extweather") |  | ||||||
|                 } |                 } | ||||||
|  |  | ||||||
|  |                 ButtonSelector { | ||||||
|  |                     value: i18n("Weather") | ||||||
|  |                     onButtonActivated: awKeys.editItem("extweather") | ||||||
|                 } |                 } | ||||||
|             } |             } | ||||||
|         } |         } | ||||||
| @ -342,21 +252,13 @@ Item { | |||||||
|  |  | ||||||
|         // update hdd model |         // update hdd model | ||||||
|         hdd.model = awKeys.getHddDevices() |         hdd.model = awKeys.getHddDevices() | ||||||
|         for (var i=0; i<hdd.model.length; i++) { |         hdd.onCompleted | ||||||
|             if (hdd.model[i] == cfg_dataengine["HDDDEV"]) { |  | ||||||
|                 if (debug) console.info("Found", hdd.model[i], "on", i) |  | ||||||
|                 hdd.currentIndex = i; |  | ||||||
|             } |  | ||||||
|         } |  | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     Component.onDestruction: { |     Component.onDestruction: { | ||||||
|         if (debug) console.debug() |         if (debug) console.debug() | ||||||
|  |  | ||||||
|         cfg_dataengine["GPUDEV"] = gpuDev.currentText |         cfg_dataengine["MPRIS"] = mpris.editText | ||||||
|         cfg_dataengine["HDDDEV"] = hdd.currentText |  | ||||||
|         cfg_dataengine["PLAYER"] = player.currentText |  | ||||||
|         cfg_dataengine["MPRIS"] = mpris.currentText |  | ||||||
|         awConfig.writeDataEngineConfiguration(cfg_dataengine) |         awConfig.writeDataEngineConfiguration(cfg_dataengine) | ||||||
|     } |     } | ||||||
| } | } | ||||||
|  | |||||||
| @ -36,6 +36,12 @@ Item { | |||||||
|     AWActions { |     AWActions { | ||||||
|         id: awActions |         id: awActions | ||||||
|     } |     } | ||||||
|  |     AWTelemetryHandler { | ||||||
|  |         id: awTelemetryHandler | ||||||
|  |     } | ||||||
|  |     BugReport { | ||||||
|  |         id: bugReport | ||||||
|  |     } | ||||||
|  |  | ||||||
|     property bool debug: awActions.isDebugEnabled() |     property bool debug: awActions.isDebugEnabled() | ||||||
|     property variant tooltipSettings: { |     property variant tooltipSettings: { | ||||||
| @ -46,15 +52,15 @@ Item { | |||||||
|         "cpuclTooltip": plasmoid.configuration.cpuclTooltip, |         "cpuclTooltip": plasmoid.configuration.cpuclTooltip, | ||||||
|         "memTooltip": plasmoid.configuration.memTooltip, |         "memTooltip": plasmoid.configuration.memTooltip, | ||||||
|         "swapTooltip": plasmoid.configuration.swapTooltip, |         "swapTooltip": plasmoid.configuration.swapTooltip, | ||||||
|         "downTooltip": plasmoid.configuration.downTooltip, |         "downkbTooltip": plasmoid.configuration.downkbTooltip, | ||||||
|         "upTooltip": plasmoid.configuration.downTooltip, |         "upkbTooltip": plasmoid.configuration.downkbTooltip, | ||||||
|         "batTooltip": plasmoid.configuration.batTooltip, |         "batTooltip": plasmoid.configuration.batTooltip, | ||||||
|         "cpuTooltipColor": plasmoid.configuration.cpuTooltipColor, |         "cpuTooltipColor": plasmoid.configuration.cpuTooltipColor, | ||||||
|         "cpuclTooltipColor": plasmoid.configuration.cpuclTooltipColor, |         "cpuclTooltipColor": plasmoid.configuration.cpuclTooltipColor, | ||||||
|         "memTooltipColor": plasmoid.configuration.memTooltipColor, |         "memTooltipColor": plasmoid.configuration.memTooltipColor, | ||||||
|         "swapTooltipColor": plasmoid.configuration.swapTooltipColor, |         "swapTooltipColor": plasmoid.configuration.swapTooltipColor, | ||||||
|         "downTooltipColor": plasmoid.configuration.downTooltipColor, |         "downkbTooltipColor": plasmoid.configuration.downkbTooltipColor, | ||||||
|         "upTooltipColor": plasmoid.configuration.upTooltipColor, |         "upkbTooltipColor": plasmoid.configuration.upkbTooltipColor, | ||||||
|         "batTooltipColor": plasmoid.configuration.batTooltipColor, |         "batTooltipColor": plasmoid.configuration.batTooltipColor, | ||||||
|         "batInTooltipColor": plasmoid.configuration.batInTooltipColor, |         "batInTooltipColor": plasmoid.configuration.batInTooltipColor, | ||||||
|         // additional field to parse AC status |         // additional field to parse AC status | ||||||
| @ -89,14 +95,17 @@ Item { | |||||||
|         textFormat: Text.RichText |         textFormat: Text.RichText | ||||||
|         wrapMode: plasmoid.configuration.wrapText ? Text.WordWrap : Text.NoWrap |         wrapMode: plasmoid.configuration.wrapText ? Text.WordWrap : Text.NoWrap | ||||||
|  |  | ||||||
|         horizontalAlignment: general.align[plasmoid.configuration.textAlign] |         horizontalAlignment: General.align[plasmoid.configuration.textAlign] | ||||||
|         verticalAlignment: Text.AlignVCenter |         verticalAlignment: Text.AlignVCenter | ||||||
|  |  | ||||||
|         color: plasmoid.configuration.fontColor |         color: plasmoid.configuration.fontColor | ||||||
|         font.family: plasmoid.configuration.fontFamily |         font.family: plasmoid.configuration.fontFamily | ||||||
|         font.italic: plasmoid.configuration.fontStyle == "italic" ? true : false |         font.italic: plasmoid.configuration.fontStyle == "italic" ? true : false | ||||||
|         font.pointSize: plasmoid.configuration.fontSize |         font.pointSize: plasmoid.configuration.fontSize | ||||||
|         font.weight: general.fontWeight[plasmoid.configuration.fontWeight] |         font.weight: General.fontWeight[plasmoid.configuration.fontWeight] | ||||||
|  |  | ||||||
|  |         style: General.textStyle[plasmoid.configuration.textStyle] | ||||||
|  |         styleColor: plasmoid.configuration.textStyleColor | ||||||
|  |  | ||||||
|         PlasmaCore.ToolTipArea { |         PlasmaCore.ToolTipArea { | ||||||
|             height: tooltip.height |             height: tooltip.height | ||||||
| @ -136,9 +145,12 @@ Item { | |||||||
|         if (debug) console.debug() |         if (debug) console.debug() | ||||||
|  |  | ||||||
|         // actions |         // actions | ||||||
|  |         // it makes no sense to use this field with optimization enable | ||||||
|  |         if (!plasmoid.configuration.optimize) | ||||||
|             plasmoid.setAction("requestKey", i18n("Request key"), "utilities-system-monitor") |             plasmoid.setAction("requestKey", i18n("Request key"), "utilities-system-monitor") | ||||||
|         plasmoid.setAction("showReadme", i18n("Show README"), "text-x-readme") |         plasmoid.setAction("showReadme", i18n("Show README"), "text-x-readme") | ||||||
|         plasmoid.setAction("checkUpdates", i18n("Check updates"), "system-software-update") |         plasmoid.setAction("checkUpdates", i18n("Check updates"), "system-software-update") | ||||||
|  |         plasmoid.setAction("reportBug", i18n("Report bug"), "tools-report-bug") | ||||||
|         // init submodule |         // init submodule | ||||||
|         Plasmoid.userConfiguringChanged(false) |         Plasmoid.userConfiguringChanged(false) | ||||||
|         // connect data |         // connect data | ||||||
| @ -187,9 +199,9 @@ Item { | |||||||
|         if (debug) console.debug() |         if (debug) console.debug() | ||||||
|  |  | ||||||
|         // init submodule |         // init submodule | ||||||
|         awKeys.initKeys(plasmoid.configuration.text, plasmoid.configuration.interval, |  | ||||||
|                         plasmoid.configuration.queueLimit) |  | ||||||
|         awKeys.initDataAggregator(tooltipSettings) |         awKeys.initDataAggregator(tooltipSettings) | ||||||
|  |         awKeys.initKeys(plasmoid.configuration.text, plasmoid.configuration.interval, | ||||||
|  |                         plasmoid.configuration.queueLimit, plasmoid.configuration.optimize) | ||||||
|         awKeys.setWrapNewLines(plasmoid.configuration.wrapNewLines) |         awKeys.setWrapNewLines(plasmoid.configuration.wrapNewLines) | ||||||
|         // configure aggregator |         // configure aggregator | ||||||
|         awKeys.setAggregatorProperty("acOffline", plasmoid.configuration.acOffline) |         awKeys.setAggregatorProperty("acOffline", plasmoid.configuration.acOffline) | ||||||
| @ -198,6 +210,15 @@ Item { | |||||||
|         awKeys.setAggregatorProperty("customUptime", plasmoid.configuration.customUptime) |         awKeys.setAggregatorProperty("customUptime", plasmoid.configuration.customUptime) | ||||||
|         awKeys.setAggregatorProperty("tempUnits", plasmoid.configuration.tempUnits) |         awKeys.setAggregatorProperty("tempUnits", plasmoid.configuration.tempUnits) | ||||||
|         awKeys.setAggregatorProperty("translate", plasmoid.configuration.translateStrings) |         awKeys.setAggregatorProperty("translate", plasmoid.configuration.translateStrings) | ||||||
|  |         // update telemetry ID | ||||||
|  |         if (plasmoid.configuration.telemetryId.length == 0) | ||||||
|  |             plasmoid.configuration.telemetryId = generateUuid() | ||||||
|  |         // save telemetry | ||||||
|  |         awTelemetryHandler.init(plasmoid.configuration.telemetryCount, | ||||||
|  |                                 plasmoid.configuration.telemetryRemote, | ||||||
|  |                                 plasmoid.configuration.telemetryId) | ||||||
|  |         if (awTelemetryHandler.put("awwidgetconfig", plasmoid.configuration.text)) | ||||||
|  |             awTelemetryHandler.uploadTelemetry("awwidgetconfig", plasmoid.configuration.text) | ||||||
|     } |     } | ||||||
|  |  | ||||||
|  |  | ||||||
| @ -213,10 +234,25 @@ Item { | |||||||
|         return awActions.showReadme() |         return awActions.showReadme() | ||||||
|     } |     } | ||||||
|  |  | ||||||
|  |     function action_reportBug() { | ||||||
|  |         if (debug) console.debug() | ||||||
|  |  | ||||||
|  |         bugReport.reset() | ||||||
|  |         bugReport.open() | ||||||
|  |     } | ||||||
|  |  | ||||||
|     function action_requestKey() { |     function action_requestKey() { | ||||||
|         if (debug) console.debug() |         if (debug) console.debug() | ||||||
|  |  | ||||||
|         tagSelectorBox.model = awKeys.dictKeys(true) |         tagSelectorBox.model = awKeys.dictKeys(true) | ||||||
|         return tagSelector.open() |         return tagSelector.open() | ||||||
|     } |     } | ||||||
|  |  | ||||||
|  |     // code from http://stackoverflow.com/questions/105034/create-guid-uuid-in-javascript | ||||||
|  |     function generateUuid() { | ||||||
|  |         return 'xxxxxxxx-xxxx-4xxx-yxxx-xxxxxxxxxxxx'.replace(/[xy]/g, function(c) { | ||||||
|  |             var r = Math.random() * 16 | 0, v = c == 'x' ? r : (r & 0x3 | 0x8); | ||||||
|  |             return v.toString(16); | ||||||
|  |         }); | ||||||
|  |     } | ||||||
| } | } | ||||||
|  | |||||||
| @ -1 +1,27 @@ | |||||||
| singleton general 1.0 general.qml | # Do not edit qmldir directly it will be overrided during compilation, | ||||||
|  | # edit qml/qmldir.in file instead. | ||||||
|  |  | ||||||
|  |  | ||||||
|  | # common QML constants | ||||||
|  | singleton General 1.0 file:///usr/share/awesomewidgets/qml/General.qml | ||||||
|  |  | ||||||
|  | # custom QML UI classes | ||||||
|  | AboutTab file:///usr/share/awesomewidgets/qml/AboutTab.qml | ||||||
|  | AWExtensions file:///usr/share/awesomewidgets/qml/AWExtensions.qml | ||||||
|  | AWInfoLabel file:///usr/share/awesomewidgets/qml/AWInfoLabel.qml | ||||||
|  | AWTagSelector file:///usr/share/awesomewidgets/qml/AWTagSelector.qml | ||||||
|  | AWTextEditor file:///usr/share/awesomewidgets/qml/AWTextEditor.qml | ||||||
|  | BugReport file:///usr/share/awesomewidgets/qml/BugReport.qml | ||||||
|  | ButtonSelector file:///usr/share/awesomewidgets/qml/ButtonSelector.qml | ||||||
|  | CheckBoxSelector file:///usr/share/awesomewidgets/qml/CheckBoxSelector.qml | ||||||
|  | ColorSelector file:///usr/share/awesomewidgets/qml/ColorSelector.qml | ||||||
|  | ComboBoxSelector file:///usr/share/awesomewidgets/qml/ComboBoxSelector.qml | ||||||
|  | ExportDialog file:///usr/share/awesomewidgets/qml/ExportDialog.qml | ||||||
|  | FontSelector file:///usr/share/awesomewidgets/qml/FontSelector.qml | ||||||
|  | HtmlDefaultFunctionsBar file:///usr/share/awesomewidgets/qml/HtmlDefaultFunctionsBar.qml | ||||||
|  | HtmlEditorButton file:///usr/share/awesomewidgets/qml/HtmlEditorButton.qml | ||||||
|  | HtmlEditorColor file:///usr/share/awesomewidgets/qml/HtmlEditorColor.qml | ||||||
|  | HtmlEditorFont file:///usr/share/awesomewidgets/qml/HtmlEditorFont.qml | ||||||
|  | ImportDialog file:///usr/share/awesomewidgets/qml/ImportDialog.qml | ||||||
|  | IntegerSelector file:///usr/share/awesomewidgets/qml/IntegerSelector.qml | ||||||
|  | LineSelector file:///usr/share/awesomewidgets/qml/LineSelector.qml | ||||||
|  | |||||||
| @ -17,8 +17,6 @@ | |||||||
|  |  | ||||||
| import QtQuick 2.0 | import QtQuick 2.0 | ||||||
| import QtQuick.Controls 1.3 as QtControls | import QtQuick.Controls 1.3 as QtControls | ||||||
| import QtQuick.Controls.Styles 1.3 as QtStyles |  | ||||||
| import QtQuick.Dialogs 1.1 as QtDialogs |  | ||||||
|  |  | ||||||
| import org.kde.plasma.private.awesomewidget 1.0 | import org.kde.plasma.private.awesomewidget 1.0 | ||||||
|  |  | ||||||
| @ -39,21 +37,21 @@ Item { | |||||||
|  |  | ||||||
|     property alias cfg_tooltipNumber: tooltipNumber.value |     property alias cfg_tooltipNumber: tooltipNumber.value | ||||||
|     property alias cfg_useTooltipBackground: useTooltipBackground.checked |     property alias cfg_useTooltipBackground: useTooltipBackground.checked | ||||||
|     property alias cfg_tooltipBackground: tooltipBackground.text |     property alias cfg_tooltipBackground: tooltipBackground.value | ||||||
|     property alias cfg_cpuTooltip: cpuTooltip.checked |     property alias cfg_cpuTooltip: cpuTooltip.checked | ||||||
|     property alias cfg_cpuTooltipColor: cpuTooltipColor.text |     property alias cfg_cpuTooltipColor: cpuTooltipColor.value | ||||||
|     property alias cfg_cpuclTooltip: cpuclTooltip.checked |     property alias cfg_cpuclTooltip: cpuclTooltip.checked | ||||||
|     property alias cfg_cpuclTooltipColor: cpuclTooltipColor.text |     property alias cfg_cpuclTooltipColor: cpuclTooltipColor.value | ||||||
|     property alias cfg_memTooltip: memTooltip.checked |     property alias cfg_memTooltip: memTooltip.checked | ||||||
|     property alias cfg_memTooltipColor: memTooltipColor.text |     property alias cfg_memTooltipColor: memTooltipColor.value | ||||||
|     property alias cfg_swapTooltip: swapTooltip.checked |     property alias cfg_swapTooltip: swapTooltip.checked | ||||||
|     property alias cfg_swapTooltipColor: swapTooltipColor.text |     property alias cfg_swapTooltipColor: swapTooltipColor.value | ||||||
|     property alias cfg_downTooltip: downTooltip.checked |     property alias cfg_downkbTooltip: downkbTooltip.checked | ||||||
|     property alias cfg_downTooltipColor: downTooltipColor.text |     property alias cfg_downkbTooltipColor: downkbTooltipColor.value | ||||||
|     property alias cfg_upTooltipColor: upTooltipColor.text |     property alias cfg_upkbTooltipColor: upkbTooltipColor.value | ||||||
|     property alias cfg_batTooltip: batTooltip.checked |     property alias cfg_batTooltip: batTooltip.checked | ||||||
|     property alias cfg_batTooltipColor: batTooltipColor.text |     property alias cfg_batTooltipColor: batTooltipColor.value | ||||||
|     property alias cfg_batInTooltipColor: batInTooltipColor.text |     property alias cfg_batInTooltipColor: batInTooltipColor.value | ||||||
|  |  | ||||||
|  |  | ||||||
|     Column { |     Column { | ||||||
| @ -67,25 +65,14 @@ Item { | |||||||
|           text: i18n("CPU, CPU clock, memory, swap and network labels support graphical tooltip. To enable them just make needed checkbox checked.") |           text: i18n("CPU, CPU clock, memory, swap and network labels support graphical tooltip. To enable them just make needed checkbox checked.") | ||||||
|         } |         } | ||||||
|  |  | ||||||
|         Row { |         IntegerSelector { | ||||||
|             height: implicitHeight |  | ||||||
|             width: parent.width |  | ||||||
|             QtControls.Label { |  | ||||||
|                 height: parent.height |  | ||||||
|                 width: parent.width * 2 / 5 |  | ||||||
|                 horizontalAlignment: Text.AlignRight |  | ||||||
|                 verticalAlignment: Text.AlignVCenter |  | ||||||
|                 text: i18n("Number of values for tooltips") |  | ||||||
|             } |  | ||||||
|             QtControls.SpinBox { |  | ||||||
|             id: tooltipNumber |             id: tooltipNumber | ||||||
|                 width: parent.width * 3 / 5 |  | ||||||
|                 minimumValue: 50 |  | ||||||
|             maximumValue: 1000 |             maximumValue: 1000 | ||||||
|  |             minimumValue: 50 | ||||||
|             stepSize: 25 |             stepSize: 25 | ||||||
|  |             text: i18n("Number of values for tooltips") | ||||||
|             value: plasmoid.configuration.tooltipNumber |             value: plasmoid.configuration.tooltipNumber | ||||||
|         } |         } | ||||||
|         } |  | ||||||
|  |  | ||||||
|         QtControls.GroupBox { |         QtControls.GroupBox { | ||||||
|             id: useTooltipBackground |             id: useTooltipBackground | ||||||
| @ -93,34 +80,10 @@ Item { | |||||||
|             width: parent.width |             width: parent.width | ||||||
|             checkable: true |             checkable: true | ||||||
|             title: i18n("Background") |             title: i18n("Background") | ||||||
|             Row { |             ColorSelector { | ||||||
|                 height: implicitHeight |  | ||||||
|                 width: parent.width |  | ||||||
|                 QtControls.Label { |  | ||||||
|                     height: parent.height |  | ||||||
|                     width: parent.width * 2 / 5 |  | ||||||
|                     horizontalAlignment: Text.AlignRight |  | ||||||
|                     verticalAlignment: Text.AlignVCenter |  | ||||||
|                     text: i18n("Background color") |  | ||||||
|                 } |  | ||||||
|                 QtControls.Button { |  | ||||||
|                 id: tooltipBackground |                 id: tooltipBackground | ||||||
|                     width: parent.width * 3 / 5 |                 text: i18n("Background color") | ||||||
|                     style: QtStyles.ButtonStyle { |                 value: plasmoid.configuration.tooltipBackground | ||||||
|                         background: Rectangle { |  | ||||||
|                             color: plasmoid.configuration.tooltipBackground |  | ||||||
|                         } |  | ||||||
|                     } |  | ||||||
|                     text: plasmoid.configuration.tooltipBackground |  | ||||||
|                     onClicked: tooltipBackgroundDialog.visible = true |  | ||||||
|                 } |  | ||||||
|  |  | ||||||
|                 QtDialogs.ColorDialog { |  | ||||||
|                     id: tooltipBackgroundDialog |  | ||||||
|                     title: i18n("Select a color") |  | ||||||
|                     color: tooltipBackground.text |  | ||||||
|                     onAccepted: tooltipBackground.text = tooltipBackgroundDialog.color |  | ||||||
|                 } |  | ||||||
|             } |             } | ||||||
|         } |         } | ||||||
|  |  | ||||||
| @ -130,34 +93,10 @@ Item { | |||||||
|             width: parent.width |             width: parent.width | ||||||
|             checkable: true |             checkable: true | ||||||
|             title: i18n("CPU") |             title: i18n("CPU") | ||||||
|             Row { |             ColorSelector { | ||||||
|                 height: implicitHeight |  | ||||||
|                 width: parent.width |  | ||||||
|                 QtControls.Label { |  | ||||||
|                     height: parent.height |  | ||||||
|                     width: parent.width * 2 / 5 |  | ||||||
|                     horizontalAlignment: Text.AlignRight |  | ||||||
|                     verticalAlignment: Text.AlignVCenter |  | ||||||
|                     text: i18n("CPU color") |  | ||||||
|                 } |  | ||||||
|                 QtControls.Button { |  | ||||||
|                 id: cpuTooltipColor |                 id: cpuTooltipColor | ||||||
|                     width: parent.width * 3 / 5 |                 text: i18n("CPU color") | ||||||
|                     style: QtStyles.ButtonStyle { |                 value: plasmoid.configuration.cpuTooltipColor | ||||||
|                         background: Rectangle { |  | ||||||
|                             color: plasmoid.configuration.cpuTooltipColor |  | ||||||
|                         } |  | ||||||
|                     } |  | ||||||
|                     text: plasmoid.configuration.cpuTooltipColor |  | ||||||
|                     onClicked: cpuTooltipColorDialog.visible = true |  | ||||||
|                 } |  | ||||||
|  |  | ||||||
|                 QtDialogs.ColorDialog { |  | ||||||
|                     id: cpuTooltipColorDialog |  | ||||||
|                     title: i18n("Select a color") |  | ||||||
|                     color: cpuTooltipColor.text |  | ||||||
|                     onAccepted: cpuTooltipColor.text = cpuTooltipColorDialog.color |  | ||||||
|                 } |  | ||||||
|             } |             } | ||||||
|         } |         } | ||||||
|  |  | ||||||
| @ -167,34 +106,10 @@ Item { | |||||||
|             width: parent.width |             width: parent.width | ||||||
|             checkable: true |             checkable: true | ||||||
|             title: i18n("CPU clock") |             title: i18n("CPU clock") | ||||||
|             Row { |             ColorSelector { | ||||||
|                 height: implicitHeight |  | ||||||
|                 width: parent.width |  | ||||||
|                 QtControls.Label { |  | ||||||
|                     height: parent.height |  | ||||||
|                     width: parent.width * 2 / 5 |  | ||||||
|                     horizontalAlignment: Text.AlignRight |  | ||||||
|                     verticalAlignment: Text.AlignVCenter |  | ||||||
|                     text: i18n("CPU clock color") |  | ||||||
|                 } |  | ||||||
|                 QtControls.Button { |  | ||||||
|                 id: cpuclTooltipColor |                 id: cpuclTooltipColor | ||||||
|                     width: parent.width * 3 / 5 |                 text: i18n("CPU clock color") | ||||||
|                     style: QtStyles.ButtonStyle { |                 value: plasmoid.configuration.cpuclTooltipColor | ||||||
|                         background: Rectangle { |  | ||||||
|                             color: plasmoid.configuration.cpuclTooltipColor |  | ||||||
|                         } |  | ||||||
|                     } |  | ||||||
|                     text: plasmoid.configuration.cpuclTooltipColor |  | ||||||
|                     onClicked: cpuclTooltipColorDialog.visible = true |  | ||||||
|                 } |  | ||||||
|  |  | ||||||
|                 QtDialogs.ColorDialog { |  | ||||||
|                     id: cpuclTooltipColorDialog |  | ||||||
|                     title: i18n("Select a color") |  | ||||||
|                     color: cpuclTooltipColor.text |  | ||||||
|                     onAccepted: cpuclTooltipColor.text = cpuclTooltipColorDialog.color |  | ||||||
|                 } |  | ||||||
|             } |             } | ||||||
|         } |         } | ||||||
|  |  | ||||||
| @ -204,34 +119,10 @@ Item { | |||||||
|             width: parent.width |             width: parent.width | ||||||
|             checkable: true |             checkable: true | ||||||
|             title: i18n("Memory") |             title: i18n("Memory") | ||||||
|             Row { |             ColorSelector { | ||||||
|                 height: implicitHeight |  | ||||||
|                 width: parent.width |  | ||||||
|                 QtControls.Label { |  | ||||||
|                     height: parent.height |  | ||||||
|                     width: parent.width * 2 / 5 |  | ||||||
|                     horizontalAlignment: Text.AlignRight |  | ||||||
|                     verticalAlignment: Text.AlignVCenter |  | ||||||
|                     text: i18n("Memory color") |  | ||||||
|                 } |  | ||||||
|                 QtControls.Button { |  | ||||||
|                 id: memTooltipColor |                 id: memTooltipColor | ||||||
|                     width: parent.width * 3 / 5 |                 text: i18n("Memory color") | ||||||
|                     style: QtStyles.ButtonStyle { |                 value: plasmoid.configuration.memTooltipColor | ||||||
|                         background: Rectangle { |  | ||||||
|                             color: plasmoid.configuration.memTooltipColor |  | ||||||
|                         } |  | ||||||
|                     } |  | ||||||
|                     text: plasmoid.configuration.memTooltipColor |  | ||||||
|                     onClicked: memTooltipColorDialog.visible = true |  | ||||||
|                 } |  | ||||||
|  |  | ||||||
|                 QtDialogs.ColorDialog { |  | ||||||
|                     id: memTooltipColorDialog |  | ||||||
|                     title: i18n("Select a color") |  | ||||||
|                     color: memTooltipColor.text |  | ||||||
|                     onAccepted: memTooltipColor.text = memTooltipColorDialog.color |  | ||||||
|                 } |  | ||||||
|             } |             } | ||||||
|         } |         } | ||||||
|  |  | ||||||
| @ -241,39 +132,15 @@ Item { | |||||||
|             width: parent.width |             width: parent.width | ||||||
|             checkable: true |             checkable: true | ||||||
|             title: i18n("Swap") |             title: i18n("Swap") | ||||||
|             Row { |             ColorSelector { | ||||||
|                 height: implicitHeight |  | ||||||
|                 width: parent.width |  | ||||||
|                 QtControls.Label { |  | ||||||
|                     height: parent.height |  | ||||||
|                     width: parent.width * 2 / 5 |  | ||||||
|                     horizontalAlignment: Text.AlignRight |  | ||||||
|                     verticalAlignment: Text.AlignVCenter |  | ||||||
|                     text: i18n("Swap color") |  | ||||||
|                 } |  | ||||||
|                 QtControls.Button { |  | ||||||
|                 id: swapTooltipColor |                 id: swapTooltipColor | ||||||
|                     width: parent.width * 3 / 5 |                 text: i18n("Swap color") | ||||||
|                     style: QtStyles.ButtonStyle { |                 value: plasmoid.configuration.swapTooltipColor | ||||||
|                         background: Rectangle { |  | ||||||
|                             color: plasmoid.configuration.swapTooltipColor |  | ||||||
|                         } |  | ||||||
|                     } |  | ||||||
|                     text: plasmoid.configuration.swapTooltipColor |  | ||||||
|                     onClicked: swapTooltipColorDialog.visible = true |  | ||||||
|                 } |  | ||||||
|  |  | ||||||
|                 QtDialogs.ColorDialog { |  | ||||||
|                     id: swapTooltipColorDialog |  | ||||||
|                     title: i18n("Select a color") |  | ||||||
|                     color: swapTooltipColor.text |  | ||||||
|                     onAccepted: swapTooltipColor.text = swapTooltipColorDialog.color |  | ||||||
|                 } |  | ||||||
|             } |             } | ||||||
|         } |         } | ||||||
|  |  | ||||||
|         QtControls.GroupBox { |         QtControls.GroupBox { | ||||||
|             id: downTooltip |             id: downkbTooltip | ||||||
|             height: implicitHeight |             height: implicitHeight | ||||||
|             width: parent.width |             width: parent.width | ||||||
|             checkable: true |             checkable: true | ||||||
| @ -281,63 +148,15 @@ Item { | |||||||
|             Column { |             Column { | ||||||
|                 height: implicitHeight |                 height: implicitHeight | ||||||
|                 width: parent.width |                 width: parent.width | ||||||
|                 Row { |                 ColorSelector { | ||||||
|                     height: implicitHeight |                     id: downkbTooltipColor | ||||||
|                     width: parent.width |  | ||||||
|                     QtControls.Label { |  | ||||||
|                         height: parent.height |  | ||||||
|                         width: parent.width * 2 / 5 |  | ||||||
|                         horizontalAlignment: Text.AlignRight |  | ||||||
|                         verticalAlignment: Text.AlignVCenter |  | ||||||
|                     text: i18n("Download speed color") |                     text: i18n("Download speed color") | ||||||
|  |                     value: plasmoid.configuration.downkbTooltipColor | ||||||
|                 } |                 } | ||||||
|                     QtControls.Button { |                 ColorSelector { | ||||||
|                         id: downTooltipColor |                     id: upkbTooltipColor | ||||||
|                         width: parent.width * 3 / 5 |  | ||||||
|                         style: QtStyles.ButtonStyle { |  | ||||||
|                             background: Rectangle { |  | ||||||
|                                 color: plasmoid.configuration.downTooltipColor |  | ||||||
|                             } |  | ||||||
|                         } |  | ||||||
|                         text: plasmoid.configuration.downTooltipColor |  | ||||||
|                         onClicked: downTooltipColorDialog.visible = true |  | ||||||
|                     } |  | ||||||
|  |  | ||||||
|                     QtDialogs.ColorDialog { |  | ||||||
|                         id: downTooltipColorDialog |  | ||||||
|                         title: i18n("Select a color") |  | ||||||
|                         color: downTooltipColor.text |  | ||||||
|                         onAccepted: downTooltipColor.text = downTooltipColorDialog.color |  | ||||||
|                     } |  | ||||||
|                 } |  | ||||||
|                 Row { |  | ||||||
|                     height: implicitHeight |  | ||||||
|                     width: parent.width |  | ||||||
|                     QtControls.Label { |  | ||||||
|                         height: parent.height |  | ||||||
|                         width: parent.width * 2 / 5 |  | ||||||
|                         horizontalAlignment: Text.AlignRight |  | ||||||
|                         verticalAlignment: Text.AlignVCenter |  | ||||||
|                     text: i18n("Upload speed color") |                     text: i18n("Upload speed color") | ||||||
|                     } |                     value: plasmoid.configuration.upkbTooltipColor | ||||||
|                     QtControls.Button { |  | ||||||
|                         id: upTooltipColor |  | ||||||
|                         width: parent.width * 3 / 5 |  | ||||||
|                         style: QtStyles.ButtonStyle { |  | ||||||
|                             background: Rectangle { |  | ||||||
|                                 color: plasmoid.configuration.upTooltipColor |  | ||||||
|                             } |  | ||||||
|                         } |  | ||||||
|                         text: plasmoid.configuration.upTooltipColor |  | ||||||
|                         onClicked: upTooltipColorDialog.visible = true |  | ||||||
|                     } |  | ||||||
|  |  | ||||||
|                     QtDialogs.ColorDialog { |  | ||||||
|                         id: upTooltipColorDialog |  | ||||||
|                         title: i18n("Select a color") |  | ||||||
|                         color: upTooltipColor.text |  | ||||||
|                         onAccepted: upTooltipColor.text = upTooltipColorDialog.color |  | ||||||
|                     } |  | ||||||
|                 } |                 } | ||||||
|             } |             } | ||||||
|         } |         } | ||||||
| @ -351,63 +170,15 @@ Item { | |||||||
|             Column { |             Column { | ||||||
|                 height: implicitHeight |                 height: implicitHeight | ||||||
|                 width: parent.width |                 width: parent.width | ||||||
|                 Row { |                 ColorSelector { | ||||||
|                     height: implicitHeight |  | ||||||
|                     width: parent.width |  | ||||||
|                     QtControls.Label { |  | ||||||
|                         height: parent.height |  | ||||||
|                         width: parent.width * 2 / 5 |  | ||||||
|                         horizontalAlignment: Text.AlignRight |  | ||||||
|                         verticalAlignment: Text.AlignVCenter |  | ||||||
|                         text: i18n("Battery active color") |  | ||||||
|                     } |  | ||||||
|                     QtControls.Button { |  | ||||||
|                     id: batTooltipColor |                     id: batTooltipColor | ||||||
|                         width: parent.width * 3 / 5 |                     text: i18n("Battery active color") | ||||||
|                         style: QtStyles.ButtonStyle { |                     value: plasmoid.configuration.batTooltipColor | ||||||
|                             background: Rectangle { |  | ||||||
|                                 color: plasmoid.configuration.batTooltipColor |  | ||||||
|                 } |                 } | ||||||
|                         } |                 ColorSelector { | ||||||
|                         text: plasmoid.configuration.batTooltipColor |  | ||||||
|                         onClicked: batTooltipColorDialog.visible = true |  | ||||||
|                     } |  | ||||||
|  |  | ||||||
|                     QtDialogs.ColorDialog { |  | ||||||
|                         id: batTooltipColorDialog |  | ||||||
|                         title: i18n("Select a color") |  | ||||||
|                         color: batTooltipColor.text |  | ||||||
|                         onAccepted: batTooltipColor.text = batTooltipColorDialog.color |  | ||||||
|                     } |  | ||||||
|                 } |  | ||||||
|                 Row { |  | ||||||
|                     height: implicitHeight |  | ||||||
|                     width: parent.width |  | ||||||
|                     QtControls.Label { |  | ||||||
|                         height: parent.height |  | ||||||
|                         width: parent.width * 2 / 5 |  | ||||||
|                         horizontalAlignment: Text.AlignRight |  | ||||||
|                         verticalAlignment: Text.AlignVCenter |  | ||||||
|                         text: i18n("Battery inactive color") |  | ||||||
|                     } |  | ||||||
|                     QtControls.Button { |  | ||||||
|                     id: batInTooltipColor |                     id: batInTooltipColor | ||||||
|                         width: parent.width * 3 / 5 |                     text: i18n("Battery inactive color") | ||||||
|                         style: QtStyles.ButtonStyle { |                     value: plasmoid.configuration.batInTooltipColor | ||||||
|                             background: Rectangle { |  | ||||||
|                                 color: plasmoid.configuration.batInTooltipColor |  | ||||||
|                             } |  | ||||||
|                         } |  | ||||||
|                         text: plasmoid.configuration.batInTooltipColor |  | ||||||
|                         onClicked: batInTooltipColorDialog.visible = true |  | ||||||
|                     } |  | ||||||
|  |  | ||||||
|                     QtDialogs.ColorDialog { |  | ||||||
|                         id: batInTooltipColorDialog |  | ||||||
|                         title: i18n("Select a color") |  | ||||||
|                         color: batInTooltipColor.text |  | ||||||
|                         onAccepted: batInTooltipColor.text = batInTooltipColorDialog.color |  | ||||||
|                     } |  | ||||||
|                 } |                 } | ||||||
|             } |             } | ||||||
|         } |         } | ||||||
|  | |||||||
| @ -15,11 +15,10 @@ | |||||||
|  *   along with awesome-widgets. If not, see http://www.gnu.org/licenses/  * |  *   along with awesome-widgets. If not, see http://www.gnu.org/licenses/  * | ||||||
|  ***************************************************************************/ |  ***************************************************************************/ | ||||||
|  |  | ||||||
| import QtQuick 2.0 | import QtQuick 2.2 | ||||||
| import QtQuick.Controls 1.3 as QtControls |  | ||||||
| import QtQuick.Dialogs 1.2 as QtDialogs |  | ||||||
|  |  | ||||||
| import org.kde.plasma.private.awesomewidget 1.0 | import org.kde.plasma.private.awesomewidget 1.0 | ||||||
|  | import "." | ||||||
|  |  | ||||||
|  |  | ||||||
| Item { | Item { | ||||||
| @ -48,300 +47,33 @@ Item { | |||||||
|     Column { |     Column { | ||||||
|         id: pageColumn |         id: pageColumn | ||||||
|         anchors.fill: parent |         anchors.fill: parent | ||||||
|         QtControls.Label { |  | ||||||
|           width: parent.width |         AWInfoLabel {} | ||||||
|           horizontalAlignment: Text.AlignHCenter |  | ||||||
|           verticalAlignment: Text.AlignVCenter |         HtmlDefaultFunctionsBar { | ||||||
|           wrapMode: Text.WordWrap |             textArea: textPattern | ||||||
|           text: i18n("Detailed information may be found on <a href=\"http://arcanis.name/projects/awesome-widgets/\">project homepage</a>") |  | ||||||
|           onLinkActivated: Qt.openUrlExternally(link); |  | ||||||
|         } |         } | ||||||
|  |  | ||||||
|         Row { |         AWTagSelector { | ||||||
|             height: implicitHeight |             backend: awKeys | ||||||
|             width: parent.width |             notifyBackend: awActions | ||||||
|             QtControls.Button { |             textArea: textPattern | ||||||
|                 width: parent.width * 3 / 12 |             groups: General.awTagRegexp | ||||||
|                 text: i18n("Font") |  | ||||||
|                 iconName: "font" |  | ||||||
|  |  | ||||||
|                 onClicked: { |  | ||||||
|                     if (debug) console.debug("Font button") |  | ||||||
|                     var defaultFont = { |  | ||||||
|                         "color": plasmoid.configuration.fontColor, |  | ||||||
|                         "family": plasmoid.configuration.fontFamily, |  | ||||||
|                         "size": plasmoid.configuration.fontSize |  | ||||||
|                     } |  | ||||||
|                     var font = awActions.getFont(defaultFont) |  | ||||||
|                     var pos = textPattern.cursorPosition |  | ||||||
|                     var selected = textPattern.selectedText |  | ||||||
|                     textPattern.remove(textPattern.selectionStart, textPattern.selectionEnd) |  | ||||||
|                     textPattern.insert(pos, "<span style=\"color:" + font.color + |  | ||||||
|                                             "; font-family:'" + font.family + |  | ||||||
|                                             "'; font-size:" + font.size + "pt;\">" + |  | ||||||
|                                             selected + "</span>") |  | ||||||
|                 } |  | ||||||
|             } |  | ||||||
|             QtControls.Button { |  | ||||||
|                 width: parent.width / 12 |  | ||||||
|                 iconName: "format-indent-more" |  | ||||||
|  |  | ||||||
|                 onClicked: { |  | ||||||
|                     if (debug) console.debug("Indent button") |  | ||||||
|                     var pos = textPattern.cursorPosition |  | ||||||
|                     var selected = textPattern.selectedText |  | ||||||
|                     textPattern.remove(textPattern.selectionStart, textPattern.selectionEnd) |  | ||||||
|                     textPattern.insert(pos, selected + "<br>\n") |  | ||||||
|                 } |  | ||||||
|         } |         } | ||||||
|  |  | ||||||
|             QtControls.Button { |         AWExtensions { | ||||||
|                 width: parent.width / 12 |             id: extensions | ||||||
|                 iconName: "format-text-bold" |             backend: awKeys | ||||||
|  |             textArea: textPattern | ||||||
|                 onClicked: { |             onUnlock: lock = false | ||||||
|                     if (debug) console.debug("Bold button") |  | ||||||
|                     var pos = textPattern.cursorPosition |  | ||||||
|                     var selected = textPattern.selectedText |  | ||||||
|                     textPattern.remove(textPattern.selectionStart, textPattern.selectionEnd) |  | ||||||
|                     textPattern.insert(pos, "<b>" + selected + "</b>") |  | ||||||
|                 } |  | ||||||
|             } |  | ||||||
|             QtControls.Button { |  | ||||||
|                 width: parent.width / 12 |  | ||||||
|                 iconName: "format-text-italic" |  | ||||||
|  |  | ||||||
|                 onClicked: { |  | ||||||
|                     if (debug) console.debug("Italic button") |  | ||||||
|                     var pos = textPattern.cursorPosition |  | ||||||
|                     var selected = textPattern.selectedText |  | ||||||
|                     textPattern.remove(textPattern.selectionStart, textPattern.selectionEnd) |  | ||||||
|                     textPattern.insert(pos, "<i>" + selected + "</i>") |  | ||||||
|                 } |  | ||||||
|             } |  | ||||||
|             QtControls.Button { |  | ||||||
|                 width: parent.width / 12 |  | ||||||
|                 iconName: "format-text-underline" |  | ||||||
|  |  | ||||||
|                 onClicked: { |  | ||||||
|                     if (debug) console.debug("Underline button") |  | ||||||
|                     var pos = textPattern.cursorPosition |  | ||||||
|                     var selected = textPattern.selectedText |  | ||||||
|                     textPattern.remove(textPattern.selectionStart, textPattern.selectionEnd) |  | ||||||
|                     textPattern.insert(pos, "<u>" + selected + "</u>") |  | ||||||
|                 } |  | ||||||
|             } |  | ||||||
|             QtControls.Button { |  | ||||||
|                 width: parent.width / 12 |  | ||||||
|                 iconName: "format-text-strikethrough" |  | ||||||
|  |  | ||||||
|                 onClicked: { |  | ||||||
|                     if (debug) console.debug("Strike button") |  | ||||||
|                     var pos = textPattern.cursorPosition |  | ||||||
|                     var selected = textPattern.selectedText |  | ||||||
|                     textPattern.remove(textPattern.selectionStart, textPattern.selectionEnd) |  | ||||||
|                     textPattern.insert(pos, "<s>" + selected + "</s>") |  | ||||||
|                 } |  | ||||||
|         } |         } | ||||||
|  |  | ||||||
|             QtControls.Button { |         AWTextEditor { | ||||||
|                 width: parent.width / 12 |  | ||||||
|                 iconName: "format-justify-left" |  | ||||||
|  |  | ||||||
|                 onClicked: { |  | ||||||
|                     if (debug) console.debug("Left button") |  | ||||||
|                     var pos = textPattern.cursorPosition |  | ||||||
|                     var selected = textPattern.selectedText |  | ||||||
|                     textPattern.remove(textPattern.selectionStart, textPattern.selectionEnd) |  | ||||||
|                     textPattern.insert(pos, "<p align=\"left\">" + selected + "</p>") |  | ||||||
|                 } |  | ||||||
|             } |  | ||||||
|             QtControls.Button { |  | ||||||
|                 width: parent.width / 12 |  | ||||||
|                 iconName: "format-justify-center" |  | ||||||
|  |  | ||||||
|                 onClicked: { |  | ||||||
|                     if (debug) console.debug("Center button") |  | ||||||
|                     var pos = textPattern.cursorPosition |  | ||||||
|                     var selected = textPattern.selectedText |  | ||||||
|                     textPattern.remove(textPattern.selectionStart, textPattern.selectionEnd) |  | ||||||
|                     textPattern.insert(pos, "<p align=\"center\">" + selected + "</p>") |  | ||||||
|                 } |  | ||||||
|             } |  | ||||||
|             QtControls.Button { |  | ||||||
|                 width: parent.width / 12 |  | ||||||
|                 iconName: "format-justify-right" |  | ||||||
|  |  | ||||||
|                 onClicked: { |  | ||||||
|                     if (debug) console.debug("Right button") |  | ||||||
|                     var pos = textPattern.cursorPosition |  | ||||||
|                     var selected = textPattern.selectedText |  | ||||||
|                     textPattern.remove(textPattern.selectionStart, textPattern.selectionEnd) |  | ||||||
|                     textPattern.insert(pos, "<p align=\"right\">" + selected + "</p>") |  | ||||||
|                 } |  | ||||||
|             } |  | ||||||
|             QtControls.Button { |  | ||||||
|                 width: parent.width / 12 |  | ||||||
|                 iconName: "format-justify-fill" |  | ||||||
|  |  | ||||||
|                 onClicked: { |  | ||||||
|                     if (debug) console.debug("Justify button") |  | ||||||
|                     var pos = textPattern.cursorPosition |  | ||||||
|                     var selected = textPattern.selectedText |  | ||||||
|                     textPattern.remove(textPattern.selectionStart, textPattern.selectionEnd) |  | ||||||
|                     textPattern.insert(pos, "<p align=\"justify\">" + selected + "</p>") |  | ||||||
|                 } |  | ||||||
|             } |  | ||||||
|         } |  | ||||||
|  |  | ||||||
|         Row { |  | ||||||
|             height: implicitHeight |  | ||||||
|             width: parent.width |  | ||||||
|             QtControls.ComboBox { |  | ||||||
|                 width: parent.width * 1 / 5 |  | ||||||
|                 textRole: "label" |  | ||||||
|                 model: [ |  | ||||||
|                     { |  | ||||||
|                         'label': i18n("AC"), |  | ||||||
|                         'regexp': "^(ac|bat).*" |  | ||||||
|                     }, |  | ||||||
|                     { |  | ||||||
|                         'label': i18n("Bars"), |  | ||||||
|                         'regexp': "^bar.*" |  | ||||||
|                     }, |  | ||||||
|                     { |  | ||||||
|                         'label': i18n("CPU"), |  | ||||||
|                         'regexp': "^(cpu|gpu|la|ps|temp(?!erature)).*" |  | ||||||
|                     }, |  | ||||||
|                     { |  | ||||||
|                         'label': i18n("Desktops"), |  | ||||||
|                         'regexp': "^(n|t)?desktop(s)?" |  | ||||||
|                     }, |  | ||||||
|                     { |  | ||||||
|                         'label': i18n("HDD"), |  | ||||||
|                         'regexp': "^hdd.*" |  | ||||||
|                     }, |  | ||||||
|                     { |  | ||||||
|                         'label': i18n("Memory"), |  | ||||||
|                         'regexp': "^(mem|swap).*" |  | ||||||
|                     }, |  | ||||||
|                     { |  | ||||||
|                         'label': i18n("Network"), |  | ||||||
|                         'regexp': "^(netdev|(down|up(?!time)).*)" |  | ||||||
|                     }, |  | ||||||
|                     { |  | ||||||
|                         'label': i18n("Music player"), |  | ||||||
|                         'regexp': "(^|d|s)(album|artist|duration|progress|title)" |  | ||||||
|                     }, |  | ||||||
|                     { |  | ||||||
|                         'label': i18n("Scripts"), |  | ||||||
|                         'regexp': "^custom.*" |  | ||||||
|                     }, |  | ||||||
|                     { |  | ||||||
|                         'label': i18n("Time"), |  | ||||||
|                         'regexp': ".*time$" |  | ||||||
|                     }, |  | ||||||
|                     { |  | ||||||
|                         'label': i18n("Quotes"), |  | ||||||
|                         'regexp': "^(perc)?(ask|bid|price)(chg)?.*" |  | ||||||
|                     }, |  | ||||||
|                     { |  | ||||||
|                         'label': i18n("Upgrades"), |  | ||||||
|                         'regexp': "^pkgcount.*" |  | ||||||
|                     }, |  | ||||||
|                     { |  | ||||||
|                         'label': i18n("Weathers"), |  | ||||||
|                         'regexp': "^(weather(Id)?|humidity|pressure|temperature|timestamp)" |  | ||||||
|                     } |  | ||||||
|                 ] |  | ||||||
|                 onCurrentIndexChanged: { |  | ||||||
|                     if (debug) console.debug() |  | ||||||
|                     tags.model = awKeys.dictKeys(true, model[currentIndex]["regexp"]) |  | ||||||
|                     if (debug) console.info("Init model", tags.model, "for", model[currentIndex]["label"]) |  | ||||||
|                     tags.currentIndex = -1 |  | ||||||
|                 } |  | ||||||
|             } |  | ||||||
|             QtControls.ComboBox { |  | ||||||
|                 id: tags |  | ||||||
|                 width: parent.width * 1 / 5 |  | ||||||
|             } |  | ||||||
|             QtControls.Button { |  | ||||||
|                 width: parent.width * 1 / 5 |  | ||||||
|                 text: i18n("Add") |  | ||||||
|  |  | ||||||
|                 onClicked: { |  | ||||||
|                     if (!tags.currentText) return |  | ||||||
|                     if (debug) console.debug("Add tag button") |  | ||||||
|                     var pos = textPattern.cursorPosition |  | ||||||
|                     var selected = textPattern.selectedText |  | ||||||
|                     textPattern.remove(textPattern.selectionStart, textPattern.selectionEnd) |  | ||||||
|                     textPattern.insert(pos, selected + "$" + tags.currentText) |  | ||||||
|                 } |  | ||||||
|             } |  | ||||||
|             QtControls.Button { |  | ||||||
|                 width: parent.width * 1 / 5 |  | ||||||
|                 text: i18n("Show value") |  | ||||||
|  |  | ||||||
|                 onClicked: { |  | ||||||
|                     if (!tags.currentText) return |  | ||||||
|                     if (debug) console.debug("Show tag button") |  | ||||||
|                     var message = i18n("Tag: %1", tags.currentText) |  | ||||||
|                     message += "<br>" |  | ||||||
|                     message += i18n("Value: %1", awKeys.valueByKey(tags.currentText)) |  | ||||||
|                     message += "<br>" |  | ||||||
|                     message += i18n("Info: %1", awKeys.infoByKey(tags.currentText)) |  | ||||||
|                     awActions.sendNotification("tag", message) |  | ||||||
|                 } |  | ||||||
|             } |  | ||||||
|             QtControls.Button { |  | ||||||
|                 width: parent.width * 1 / 5 |  | ||||||
|                 text: i18n("Add lambda") |  | ||||||
|  |  | ||||||
|                 onClicked: { |  | ||||||
|                     if (debug) console.debug("Lambda button") |  | ||||||
|                     var pos = textPattern.cursorPosition |  | ||||||
|                     var selected = textPattern.selectedText |  | ||||||
|                     textPattern.remove(textPattern.selectionStart, textPattern.selectionEnd) |  | ||||||
|                     textPattern.insert(pos, selected + "${{\n\n}}") |  | ||||||
|                 } |  | ||||||
|             } |  | ||||||
|         } |  | ||||||
|  |  | ||||||
|         Row { |  | ||||||
|             height: implicitHeight |  | ||||||
|             width: parent.width |  | ||||||
|             QtControls.Button { |  | ||||||
|                 width: parent.width * 3 / 5 |  | ||||||
|                 text: i18n("Edit bars") |  | ||||||
|                 onClicked: awKeys.editItem("graphicalitem") |  | ||||||
|             } |  | ||||||
|             QtControls.Button { |  | ||||||
|                 width: parent.width * 2 / 5 |  | ||||||
|                 text: i18n("Preview") |  | ||||||
|                 onClicked: { |  | ||||||
|                     lock = false |  | ||||||
|                     awKeys.initKeys(textPattern.text, plasmoid.configuration.interval, |  | ||||||
|                                     plasmoid.configuration.queueLimit) |  | ||||||
|                     awKeys.needToBeUpdated() |  | ||||||
|                 } |  | ||||||
|             } |  | ||||||
|         } |  | ||||||
|  |  | ||||||
|         QtControls.TextArea { |  | ||||||
|             id: textPattern |             id: textPattern | ||||||
|             width: parent.width |             backend: awKeys | ||||||
|             height: parent.height * 4 / 5 |  | ||||||
|             textFormat: TextEdit.PlainText |  | ||||||
|             text: plasmoid.configuration.text |  | ||||||
|         } |         } | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     QtDialogs.MessageDialog { |  | ||||||
|         id: compiledText |  | ||||||
|         modality: Qt.NonModal |  | ||||||
|         title: i18n("Preview") |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|  |  | ||||||
|     Component.onCompleted: { |     Component.onCompleted: { | ||||||
|         if (debug) console.debug() |         if (debug) console.debug() | ||||||
| @ -349,7 +81,7 @@ Item { | |||||||
|         awKeys.needTextToBeUpdated.connect(needTextUpdate) |         awKeys.needTextToBeUpdated.connect(needTextUpdate) | ||||||
|         // init submodule |         // init submodule | ||||||
|         awKeys.initKeys(plasmoid.configuration.text, plasmoid.configuration.interval, |         awKeys.initKeys(plasmoid.configuration.text, plasmoid.configuration.interval, | ||||||
|                         plasmoid.configuration.queueLimit) |                         plasmoid.configuration.queueLimit, false) | ||||||
|         awKeys.setAggregatorProperty("acOffline", plasmoid.configuration.acOffline) |         awKeys.setAggregatorProperty("acOffline", plasmoid.configuration.acOffline) | ||||||
|         awKeys.setAggregatorProperty("acOnline", plasmoid.configuration.acOnline) |         awKeys.setAggregatorProperty("acOnline", plasmoid.configuration.acOnline) | ||||||
|         awKeys.setAggregatorProperty("customTime", plasmoid.configuration.customTime) |         awKeys.setAggregatorProperty("customTime", plasmoid.configuration.customTime) | ||||||
| @ -362,8 +94,7 @@ Item { | |||||||
|         if (lock) return |         if (lock) return | ||||||
|         if (debug) console.debug() |         if (debug) console.debug() | ||||||
|  |  | ||||||
|         compiledText.text = newText.replace(/ /g, " ") |         extensions.showMessage(newText) | ||||||
|         compiledText.open() |         lock = true | ||||||
|         lock = true; |  | ||||||
|     } |     } | ||||||
| } | } | ||||||
|  | |||||||
| @ -15,14 +15,12 @@ Icon=utilities-system-monitor | |||||||
| X-KDE-ServiceTypes=Plasma/Applet | X-KDE-ServiceTypes=Plasma/Applet | ||||||
| X-Plasma-API=declarativeappletscript | X-Plasma-API=declarativeappletscript | ||||||
| X-Plasma-MainScript=ui/main.qml | X-Plasma-MainScript=ui/main.qml | ||||||
| X-Plasma-RemoteLocation= |  | ||||||
|  |  | ||||||
| X-KDE-PluginInfo-Author=Evgeniy Alekseev aka arcanis | X-KDE-PluginInfo-Author=Evgeniy Alekseev aka arcanis | ||||||
| X-KDE-PluginInfo-Email=esalexeev@gmail.com | X-KDE-PluginInfo-Email=esalexeev@gmail.com | ||||||
| X-KDE-PluginInfo-Name=org.kde.plasma.awesomewidget | X-KDE-PluginInfo-Name=org.kde.plasma.awesomewidget | ||||||
| X-KDE-PluginInfo-Version=3.0.0 | X-KDE-PluginInfo-Version=3.5.0 | ||||||
| X-KDE-PluginInfo-Website=http://arcanis.name/projects/awesome-widgets/ | X-KDE-PluginInfo-Website=https://arcanis.me/projects/awesome-widgets/ | ||||||
| X-KDE-PluginInfo-Category=System Information | X-KDE-PluginInfo-Category=System Information | ||||||
| X-KDE-PluginInfo-Depends= |  | ||||||
| X-KDE-PluginInfo-License=GPLv3 | X-KDE-PluginInfo-License=GPLv3 | ||||||
| X-KDE-PluginInfo-EnabledByDefault=true | X-KDE-PluginInfo-EnabledByDefault=true | ||||||
|  | |||||||
| @ -4,8 +4,6 @@ add_definitions(-DTRANSLATION_DOMAIN=\"plasma_applet_org.kde.plasma.awesomewidge | |||||||
| include_directories( | include_directories( | ||||||
|         ${CMAKE_SOURCE_DIR} |         ${CMAKE_SOURCE_DIR} | ||||||
|         ${CMAKE_BINARY_DIR} |         ${CMAKE_BINARY_DIR} | ||||||
|         ${CMAKE_CURRENT_BINARY_DIR} |  | ||||||
|         ${CMAKE_CURRENT_BINARY_DIR}/../ |  | ||||||
|         ${CMAKE_CURRENT_SOURCE_DIR}/../../${PROJECT_LIBRARY}/ |         ${CMAKE_CURRENT_SOURCE_DIR}/../../${PROJECT_LIBRARY}/ | ||||||
|         ${PROJECT_TRDPARTY_DIR} |         ${PROJECT_TRDPARTY_DIR} | ||||||
|         ${Qt_INCLUDE} |         ${Qt_INCLUDE} | ||||||
| @ -13,9 +11,11 @@ include_directories( | |||||||
| ) | ) | ||||||
|  |  | ||||||
| file(GLOB SUBPROJECT_SOURCE *.cpp ${PROJECT_TRDPARTY_DIR}/fontdialog/*.cpp ${CMAKE_SOURCE_DIR}/*.cpp) | file(GLOB SUBPROJECT_SOURCE *.cpp ${PROJECT_TRDPARTY_DIR}/fontdialog/*.cpp ${CMAKE_SOURCE_DIR}/*.cpp) | ||||||
|  | file(GLOB SUBPROJECT_UI *.ui) | ||||||
| file(GLOB SUBPROJECT_NOTIFY *.notifyrc) | file(GLOB SUBPROJECT_NOTIFY *.notifyrc) | ||||||
|  |  | ||||||
| add_library(${PLUGIN_NAME} SHARED ${SUBPROJECT_SOURCE}) | qt5_wrap_ui(SUBPROJECT_UI_HEADER ${SUBPROJECT_UI}) | ||||||
|  | add_library(${PLUGIN_NAME} SHARED ${SUBPROJECT_SOURCE} ${SUBPROJECT_UI_HEADER}) | ||||||
| target_link_libraries(${PLUGIN_NAME} ${PROJECT_LIBRARY} ${Qt_LIBRARIES} ${Kf5_LIBRARIES}) | target_link_libraries(${PLUGIN_NAME} ${PROJECT_LIBRARY} ${Qt_LIBRARIES} ${Kf5_LIBRARIES}) | ||||||
|  |  | ||||||
| install(TARGETS ${PLUGIN_NAME} DESTINATION ${QML_INSTALL_DIR}/org/kde/plasma/private/awesomewidget) | install(TARGETS ${PLUGIN_NAME} DESTINATION ${QML_INSTALL_DIR}/org/kde/plasma/private/awesomewidget) | ||||||
|  | |||||||
							
								
								
									
										181
									
								
								sources/awesome-widget/plugin/awabstractpairconfig.cpp
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										181
									
								
								sources/awesome-widget/plugin/awabstractpairconfig.cpp
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,181 @@ | |||||||
|  | /*************************************************************************** | ||||||
|  |  *   This file is part of awesome-widgets                                  * | ||||||
|  |  *                                                                         * | ||||||
|  |  *   awesome-widgets is free software: you can redistribute it and/or      * | ||||||
|  |  *   modify it under the terms of the GNU General Public License as        * | ||||||
|  |  *   published by the Free Software Foundation, either version 3 of the    * | ||||||
|  |  *   License, or (at your option) any later version.                       * | ||||||
|  |  *                                                                         * | ||||||
|  |  *   awesome-widgets is distributed in the hope that it will be useful,    * | ||||||
|  |  *   but WITHOUT ANY WARRANTY; without even the implied warranty of        * | ||||||
|  |  *   MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the         * | ||||||
|  |  *   GNU General Public License for more details.                          * | ||||||
|  |  *                                                                         * | ||||||
|  |  *   You should have received a copy of the GNU General Public License     * | ||||||
|  |  *   along with awesome-widgets. If not, see http://www.gnu.org/licenses/  * | ||||||
|  |  ***************************************************************************/ | ||||||
|  |  | ||||||
|  | #include "awabstractpairconfig.h" | ||||||
|  | #include "ui_awabstractpairconfig.h" | ||||||
|  |  | ||||||
|  | #include <KI18n/KLocalizedString> | ||||||
|  |  | ||||||
|  | #include <QPushButton> | ||||||
|  | #include <utility> | ||||||
|  |  | ||||||
|  | #include "awabstractselector.h" | ||||||
|  | #include "awdebug.h" | ||||||
|  |  | ||||||
|  |  | ||||||
|  | AWAbstractPairConfig::AWAbstractPairConfig(QWidget *_parent, const bool _hasEdit, QStringList _keys) | ||||||
|  |     : QDialog(_parent) | ||||||
|  |     , ui(new Ui::AWAbstractPairConfig) | ||||||
|  |     , m_hasEdit(_hasEdit) | ||||||
|  |     , m_keys(std::move(_keys)) | ||||||
|  | { | ||||||
|  |     qCDebug(LOG_AW) << __PRETTY_FUNCTION__; | ||||||
|  |  | ||||||
|  |     ui->setupUi(this); | ||||||
|  |  | ||||||
|  |     connect(ui->buttonBox, SIGNAL(accepted()), this, SLOT(accept())); | ||||||
|  |     connect(ui->buttonBox, SIGNAL(rejected()), this, SLOT(reject())); | ||||||
|  |  | ||||||
|  |     // edit feature | ||||||
|  |     if (m_hasEdit) { | ||||||
|  |         m_editButton = ui->buttonBox->addButton(i18n("Edit"), QDialogButtonBox::ActionRole); | ||||||
|  |         connect(m_editButton, SIGNAL(clicked(bool)), this, SLOT(edit())); | ||||||
|  |     } | ||||||
|  | } | ||||||
|  |  | ||||||
|  |  | ||||||
|  | AWAbstractPairConfig::~AWAbstractPairConfig() | ||||||
|  | { | ||||||
|  |     qCDebug(LOG_AW) << __PRETTY_FUNCTION__; | ||||||
|  |  | ||||||
|  |     clearSelectors(); | ||||||
|  |  | ||||||
|  |     delete ui; | ||||||
|  | } | ||||||
|  |  | ||||||
|  |  | ||||||
|  | void AWAbstractPairConfig::showDialog() | ||||||
|  | { | ||||||
|  |     // update dialog | ||||||
|  |     updateDialog(); | ||||||
|  |     // exec dialog | ||||||
|  |     return execDialog(); | ||||||
|  | } | ||||||
|  |  | ||||||
|  |  | ||||||
|  | void AWAbstractPairConfig::setEditable(const bool _first, const bool _second) | ||||||
|  | { | ||||||
|  |     qCDebug(LOG_AW) << "Set editable" << _first << _second; | ||||||
|  |  | ||||||
|  |     m_editable = {_first, _second}; | ||||||
|  | } | ||||||
|  |  | ||||||
|  |  | ||||||
|  | void AWAbstractPairConfig::edit() | ||||||
|  | { | ||||||
|  |     m_helper->editPairs(); | ||||||
|  |     updateDialog(); | ||||||
|  | } | ||||||
|  |  | ||||||
|  |  | ||||||
|  | void AWAbstractPairConfig::updateUi() | ||||||
|  | { | ||||||
|  |     QPair<QString, QString> current = dynamic_cast<AWAbstractSelector *>(sender())->current(); | ||||||
|  |     int index = m_selectors.indexOf(dynamic_cast<AWAbstractSelector *>(sender())); | ||||||
|  |  | ||||||
|  |     if ((current.first.isEmpty()) && (current.second.isEmpty())) { | ||||||
|  |         // remove current selector if it is empty and does not last | ||||||
|  |         if (sender() == m_selectors.last()) | ||||||
|  |             return; | ||||||
|  |         AWAbstractSelector *selector = m_selectors.takeAt(index); | ||||||
|  |         ui->verticalLayout->removeWidget(selector); | ||||||
|  |         selector->deleteLater(); | ||||||
|  |     } else { | ||||||
|  |         // add new selector if something changed | ||||||
|  |         if (sender() != m_selectors.last()) | ||||||
|  |             return; | ||||||
|  |         auto keys = initKeys(); | ||||||
|  |         addSelector(keys.first, keys.second, QPair<QString, QString>()); | ||||||
|  |     } | ||||||
|  | } | ||||||
|  |  | ||||||
|  |  | ||||||
|  | void AWAbstractPairConfig::addSelector(const QStringList &_keys, const QStringList &_values, | ||||||
|  |                                        const QPair<QString, QString> &_current) | ||||||
|  | { | ||||||
|  |     qCDebug(LOG_AW) << "Add selector with keys" << _keys << "values" << _values << "and current ones" << _current; | ||||||
|  |  | ||||||
|  |     auto *selector = new AWAbstractSelector(ui->scrollAreaWidgetContents, m_editable); | ||||||
|  |     selector->init(_keys, _values, _current); | ||||||
|  |     ui->verticalLayout->insertWidget(ui->verticalLayout->count() - 1, selector); | ||||||
|  |     connect(selector, SIGNAL(selectionChanged()), this, SLOT(updateUi())); | ||||||
|  |     m_selectors.append(selector); | ||||||
|  | } | ||||||
|  |  | ||||||
|  |  | ||||||
|  | void AWAbstractPairConfig::clearSelectors() | ||||||
|  | { | ||||||
|  |     for (auto &selector : m_selectors) { | ||||||
|  |         disconnect(selector, SIGNAL(selectionChanged()), this, SLOT(updateUi())); | ||||||
|  |         ui->verticalLayout->removeWidget(selector); | ||||||
|  |         selector->deleteLater(); | ||||||
|  |     } | ||||||
|  |     m_selectors.clear(); | ||||||
|  | } | ||||||
|  |  | ||||||
|  |  | ||||||
|  | void AWAbstractPairConfig::execDialog() | ||||||
|  | { | ||||||
|  |     int ret = exec(); | ||||||
|  |     QHash<QString, QString> data; | ||||||
|  |     for (auto &selector : m_selectors) { | ||||||
|  |         QPair<QString, QString> select = selector->current(); | ||||||
|  |         if (select.first.isEmpty()) | ||||||
|  |             continue; | ||||||
|  |         data[select.first] = select.second; | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     // save configuration if required | ||||||
|  |     auto status = true; | ||||||
|  |     switch (ret) { | ||||||
|  |     case 0: | ||||||
|  |         break; | ||||||
|  |     case 1: | ||||||
|  |     default: | ||||||
|  |         status &= m_helper->writeItems(data); | ||||||
|  |         status &= m_helper->removeUnusedKeys(data.keys()); | ||||||
|  |         break; | ||||||
|  |     } | ||||||
|  |     qCDebug(LOG_AW) << "Configuration save status" << status; | ||||||
|  | } | ||||||
|  |  | ||||||
|  |  | ||||||
|  | QPair<QStringList, QStringList> AWAbstractPairConfig::initKeys() const | ||||||
|  | { | ||||||
|  |     // we are adding empty string at the start | ||||||
|  |     QStringList left = {""}; | ||||||
|  |     left.append(m_helper->leftKeys().isEmpty() ? m_keys : m_helper->leftKeys()); | ||||||
|  |     left.sort(); | ||||||
|  |     QStringList right = {""}; | ||||||
|  |     right.append(m_helper->rightKeys().isEmpty() ? m_keys : m_helper->rightKeys()); | ||||||
|  |     right.sort(); | ||||||
|  |  | ||||||
|  |     return QPair<QStringList, QStringList>(left, right); | ||||||
|  | } | ||||||
|  |  | ||||||
|  |  | ||||||
|  | void AWAbstractPairConfig::updateDialog() | ||||||
|  | { | ||||||
|  |     clearSelectors(); | ||||||
|  |     auto pairs = m_helper->pairs(); | ||||||
|  |     auto keys = initKeys(); | ||||||
|  |  | ||||||
|  |     for (auto &key : m_helper->keys()) | ||||||
|  |         addSelector(keys.first, keys.second, QPair<QString, QString>(key, m_helper->pairs()[key])); | ||||||
|  |     // empty one | ||||||
|  |     addSelector(keys.first, keys.second, QPair<QString, QString>()); | ||||||
|  | } | ||||||
							
								
								
									
										72
									
								
								sources/awesome-widget/plugin/awabstractpairconfig.h
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										72
									
								
								sources/awesome-widget/plugin/awabstractpairconfig.h
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,72 @@ | |||||||
|  | /*************************************************************************** | ||||||
|  |  *   This file is part of awesome-widgets                                  * | ||||||
|  |  *                                                                         * | ||||||
|  |  *   awesome-widgets is free software: you can redistribute it and/or      * | ||||||
|  |  *   modify it under the terms of the GNU General Public License as        * | ||||||
|  |  *   published by the Free Software Foundation, either version 3 of the    * | ||||||
|  |  *   License, or (at your option) any later version.                       * | ||||||
|  |  *                                                                         * | ||||||
|  |  *   awesome-widgets is distributed in the hope that it will be useful,    * | ||||||
|  |  *   but WITHOUT ANY WARRANTY; without even the implied warranty of        * | ||||||
|  |  *   MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the         * | ||||||
|  |  *   GNU General Public License for more details.                          * | ||||||
|  |  *                                                                         * | ||||||
|  |  *   You should have received a copy of the GNU General Public License     * | ||||||
|  |  *   along with awesome-widgets. If not, see http://www.gnu.org/licenses/  * | ||||||
|  |  ***************************************************************************/ | ||||||
|  |  | ||||||
|  |  | ||||||
|  | #ifndef AWABSTRACTPAIRCONFIG_H | ||||||
|  | #define AWABSTRACTPAIRCONFIG_H | ||||||
|  |  | ||||||
|  | #include <QDialog> | ||||||
|  |  | ||||||
|  | #include "awabstractpairhelper.h" | ||||||
|  |  | ||||||
|  |  | ||||||
|  | class AWAbstractSelector; | ||||||
|  | namespace Ui | ||||||
|  | { | ||||||
|  | class AWAbstractPairConfig; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | class AWAbstractPairConfig : public QDialog | ||||||
|  | { | ||||||
|  |     Q_OBJECT | ||||||
|  |  | ||||||
|  | public: | ||||||
|  |     explicit AWAbstractPairConfig(QWidget *_parent = nullptr, bool _hasEdit = false, QStringList _keys = QStringList()); | ||||||
|  |     ~AWAbstractPairConfig() override; | ||||||
|  |     template <class T> void initHelper() | ||||||
|  |     { | ||||||
|  |  | ||||||
|  |         delete m_helper; | ||||||
|  |         m_helper = new T(this); | ||||||
|  |     } | ||||||
|  |     void showDialog(); | ||||||
|  |     // properties | ||||||
|  |     void setEditable(bool _first, bool _second); | ||||||
|  |  | ||||||
|  | private slots: | ||||||
|  |     void edit(); | ||||||
|  |     void updateUi(); | ||||||
|  |  | ||||||
|  | private: | ||||||
|  |     QPushButton *m_editButton = nullptr; | ||||||
|  |     Ui::AWAbstractPairConfig *ui = nullptr; | ||||||
|  |     AWAbstractPairHelper *m_helper = nullptr; | ||||||
|  |     QList<AWAbstractSelector *> m_selectors; | ||||||
|  |     // properties | ||||||
|  |     QPair<bool, bool> m_editable = {false, false}; | ||||||
|  |     bool m_hasEdit = false; | ||||||
|  |     QStringList m_keys; | ||||||
|  |     // methods | ||||||
|  |     void addSelector(const QStringList &_keys, const QStringList &_values, const QPair<QString, QString> &_current); | ||||||
|  |     void clearSelectors(); | ||||||
|  |     void execDialog(); | ||||||
|  |     [[nodiscard]] QPair<QStringList, QStringList> initKeys() const; | ||||||
|  |     void updateDialog(); | ||||||
|  | }; | ||||||
|  |  | ||||||
|  |  | ||||||
|  | #endif /* AWABSTRACTPAIRCONFIG_H */ | ||||||
							
								
								
									
										93
									
								
								sources/awesome-widget/plugin/awabstractpairconfig.ui
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										93
									
								
								sources/awesome-widget/plugin/awabstractpairconfig.ui
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,93 @@ | |||||||
|  | <?xml version="1.0" encoding="UTF-8"?> | ||||||
|  | <ui version="4.0"> | ||||||
|  |  <class>AWAbstractPairConfig</class> | ||||||
|  |  <widget class="QDialog" name="AWAbstractPairConfig"> | ||||||
|  |   <property name="geometry"> | ||||||
|  |    <rect> | ||||||
|  |     <x>0</x> | ||||||
|  |     <y>0</y> | ||||||
|  |     <width>400</width> | ||||||
|  |     <height>300</height> | ||||||
|  |    </rect> | ||||||
|  |   </property> | ||||||
|  |   <layout class="QVBoxLayout" name="verticalLayout_2"> | ||||||
|  |    <item> | ||||||
|  |     <widget class="QScrollArea" name="scrollArea"> | ||||||
|  |      <property name="widgetResizable"> | ||||||
|  |       <bool>true</bool> | ||||||
|  |      </property> | ||||||
|  |      <widget class="QWidget" name="scrollAreaWidgetContents"> | ||||||
|  |       <property name="geometry"> | ||||||
|  |        <rect> | ||||||
|  |         <x>0</x> | ||||||
|  |         <y>0</y> | ||||||
|  |         <width>384</width> | ||||||
|  |         <height>249</height> | ||||||
|  |        </rect> | ||||||
|  |       </property> | ||||||
|  |       <layout class="QVBoxLayout" name="verticalLayout"> | ||||||
|  |        <item> | ||||||
|  |         <spacer name="verticalSpacer"> | ||||||
|  |          <property name="orientation"> | ||||||
|  |           <enum>Qt::Vertical</enum> | ||||||
|  |          </property> | ||||||
|  |          <property name="sizeHint" stdset="0"> | ||||||
|  |           <size> | ||||||
|  |            <width>20</width> | ||||||
|  |            <height>40</height> | ||||||
|  |           </size> | ||||||
|  |          </property> | ||||||
|  |         </spacer> | ||||||
|  |        </item> | ||||||
|  |       </layout> | ||||||
|  |      </widget> | ||||||
|  |     </widget> | ||||||
|  |    </item> | ||||||
|  |    <item> | ||||||
|  |     <widget class="QDialogButtonBox" name="buttonBox"> | ||||||
|  |      <property name="orientation"> | ||||||
|  |       <enum>Qt::Horizontal</enum> | ||||||
|  |      </property> | ||||||
|  |      <property name="standardButtons"> | ||||||
|  |       <set>QDialogButtonBox::Close|QDialogButtonBox::Ok</set> | ||||||
|  |      </property> | ||||||
|  |     </widget> | ||||||
|  |    </item> | ||||||
|  |   </layout> | ||||||
|  |  </widget> | ||||||
|  |  <resources/> | ||||||
|  |  <connections> | ||||||
|  |   <connection> | ||||||
|  |    <sender>buttonBox</sender> | ||||||
|  |    <signal>accepted()</signal> | ||||||
|  |    <receiver>AWAbstractPairConfig</receiver> | ||||||
|  |    <slot>accept()</slot> | ||||||
|  |    <hints> | ||||||
|  |     <hint type="sourcelabel"> | ||||||
|  |      <x>248</x> | ||||||
|  |      <y>254</y> | ||||||
|  |     </hint> | ||||||
|  |     <hint type="destinationlabel"> | ||||||
|  |      <x>157</x> | ||||||
|  |      <y>274</y> | ||||||
|  |     </hint> | ||||||
|  |    </hints> | ||||||
|  |   </connection> | ||||||
|  |   <connection> | ||||||
|  |    <sender>buttonBox</sender> | ||||||
|  |    <signal>rejected()</signal> | ||||||
|  |    <receiver>AWAbstractPairConfig</receiver> | ||||||
|  |    <slot>reject()</slot> | ||||||
|  |    <hints> | ||||||
|  |     <hint type="sourcelabel"> | ||||||
|  |      <x>316</x> | ||||||
|  |      <y>260</y> | ||||||
|  |     </hint> | ||||||
|  |     <hint type="destinationlabel"> | ||||||
|  |      <x>286</x> | ||||||
|  |      <y>274</y> | ||||||
|  |     </hint> | ||||||
|  |    </hints> | ||||||
|  |   </connection> | ||||||
|  |  </connections> | ||||||
|  | </ui> | ||||||
							
								
								
									
										135
									
								
								sources/awesome-widget/plugin/awabstractpairhelper.cpp
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										135
									
								
								sources/awesome-widget/plugin/awabstractpairhelper.cpp
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,135 @@ | |||||||
|  | /*************************************************************************** | ||||||
|  |  *   This file is part of awesome-widgets                                  * | ||||||
|  |  *                                                                         * | ||||||
|  |  *   awesome-widgets is free software: you can redistribute it and/or      * | ||||||
|  |  *   modify it under the terms of the GNU General Public License as        * | ||||||
|  |  *   published by the Free Software Foundation, either version 3 of the    * | ||||||
|  |  *   License, or (at your option) any later version.                       * | ||||||
|  |  *                                                                         * | ||||||
|  |  *   awesome-widgets is distributed in the hope that it will be useful,    * | ||||||
|  |  *   but WITHOUT ANY WARRANTY; without even the implied warranty of        * | ||||||
|  |  *   MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the         * | ||||||
|  |  *   GNU General Public License for more details.                          * | ||||||
|  |  *                                                                         * | ||||||
|  |  *   You should have received a copy of the GNU General Public License     * | ||||||
|  |  *   along with awesome-widgets. If not, see http://www.gnu.org/licenses/  * | ||||||
|  |  ***************************************************************************/ | ||||||
|  |  | ||||||
|  | #include "awabstractpairhelper.h" | ||||||
|  |  | ||||||
|  | #include <QSettings> | ||||||
|  | #include <QStandardPaths> | ||||||
|  | #include <utility> | ||||||
|  |  | ||||||
|  | #include "awdebug.h" | ||||||
|  |  | ||||||
|  |  | ||||||
|  | AWAbstractPairHelper::AWAbstractPairHelper(QString _filePath, QString _section) | ||||||
|  |     : m_filePath(std::move(_filePath)) | ||||||
|  |     , m_section(std::move(_section)) | ||||||
|  | { | ||||||
|  |     qCDebug(LOG_AW) << __PRETTY_FUNCTION__; | ||||||
|  |  | ||||||
|  |     AWAbstractPairHelper::initItems(); | ||||||
|  | } | ||||||
|  |  | ||||||
|  |  | ||||||
|  | AWAbstractPairHelper::~AWAbstractPairHelper() | ||||||
|  | { | ||||||
|  |     qCDebug(LOG_AW) << __PRETTY_FUNCTION__; | ||||||
|  | } | ||||||
|  |  | ||||||
|  |  | ||||||
|  | QStringList AWAbstractPairHelper::keys() const | ||||||
|  | { | ||||||
|  |     return m_pairs.keys(); | ||||||
|  | } | ||||||
|  |  | ||||||
|  |  | ||||||
|  | QHash<QString, QString> AWAbstractPairHelper::pairs() const | ||||||
|  | { | ||||||
|  |     return m_pairs; | ||||||
|  | } | ||||||
|  |  | ||||||
|  |  | ||||||
|  | QStringList AWAbstractPairHelper::values() const | ||||||
|  | { | ||||||
|  |     return m_pairs.values(); | ||||||
|  | } | ||||||
|  |  | ||||||
|  |  | ||||||
|  | QSet<QString> AWAbstractPairHelper::valuesSet() const | ||||||
|  | { | ||||||
|  |     auto values = m_pairs.values(); | ||||||
|  |     return {values.cbegin(), values.cend()}; | ||||||
|  | } | ||||||
|  |  | ||||||
|  |  | ||||||
|  | void AWAbstractPairHelper::initItems() | ||||||
|  | { | ||||||
|  |     m_pairs.clear(); | ||||||
|  |  | ||||||
|  |     QStringList configs = QStandardPaths::locateAll(QStandardPaths::GenericDataLocation, m_filePath); | ||||||
|  |  | ||||||
|  |     for (auto &fileName : configs) { | ||||||
|  |         QSettings settings(fileName, QSettings::IniFormat); | ||||||
|  |         qCInfo(LOG_AW) << "Configuration file" << settings.fileName(); | ||||||
|  |  | ||||||
|  |         settings.beginGroup(m_section); | ||||||
|  |         QStringList keys = settings.childKeys(); | ||||||
|  |         for (auto &key : keys) { | ||||||
|  |             QString value = settings.value(key).toString(); | ||||||
|  |             qCInfo(LOG_AW) << "Found key" << key << "for value" << value << "in" << settings.fileName(); | ||||||
|  |             if (value.isEmpty()) { | ||||||
|  |                 qCInfo(LOG_AW) << "Skip empty value for" << key; | ||||||
|  |                 continue; | ||||||
|  |             } | ||||||
|  |             m_pairs[key] = value; | ||||||
|  |         } | ||||||
|  |         settings.endGroup(); | ||||||
|  |     } | ||||||
|  | } | ||||||
|  |  | ||||||
|  |  | ||||||
|  | bool AWAbstractPairHelper::writeItems(const QHash<QString, QString> &_configuration) const | ||||||
|  | { | ||||||
|  |     qCDebug(LOG_AW) << "Write configuration" << _configuration; | ||||||
|  |  | ||||||
|  |     QString fileName | ||||||
|  |         = QString("%1/%2").arg(QStandardPaths::writableLocation(QStandardPaths::GenericDataLocation)).arg(m_filePath); | ||||||
|  |     QSettings settings(fileName, QSettings::IniFormat); | ||||||
|  |     qCInfo(LOG_AW) << "Configuration file" << fileName; | ||||||
|  |  | ||||||
|  |     settings.beginGroup(m_section); | ||||||
|  |     for (auto &key : _configuration.keys()) | ||||||
|  |         settings.setValue(key, _configuration[key]); | ||||||
|  |     settings.endGroup(); | ||||||
|  |  | ||||||
|  |     settings.sync(); | ||||||
|  |  | ||||||
|  |     return (settings.status() == QSettings::NoError); | ||||||
|  | } | ||||||
|  |  | ||||||
|  |  | ||||||
|  | bool AWAbstractPairHelper::removeUnusedKeys(const QStringList &_keys) const | ||||||
|  | { | ||||||
|  |     qCDebug(LOG_AW) << "Remove keys" << _keys; | ||||||
|  |  | ||||||
|  |     QString fileName | ||||||
|  |         = QString("%1/%2").arg(QStandardPaths::writableLocation(QStandardPaths::GenericDataLocation)).arg(m_filePath); | ||||||
|  |     QSettings settings(fileName, QSettings::IniFormat); | ||||||
|  |     qCInfo(LOG_AW) << "Configuration file" << fileName; | ||||||
|  |  | ||||||
|  |     settings.beginGroup(m_section); | ||||||
|  |     QStringList foundKeys = settings.childKeys(); | ||||||
|  |     for (auto &key : foundKeys) { | ||||||
|  |         if (_keys.contains(key)) | ||||||
|  |             continue; | ||||||
|  |         settings.remove(key); | ||||||
|  |     } | ||||||
|  |     settings.endGroup(); | ||||||
|  |  | ||||||
|  |     settings.sync(); | ||||||
|  |  | ||||||
|  |     return (settings.status() == QSettings::NoError); | ||||||
|  | } | ||||||
							
								
								
									
										51
									
								
								sources/awesome-widget/plugin/awabstractpairhelper.h
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										51
									
								
								sources/awesome-widget/plugin/awabstractpairhelper.h
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,51 @@ | |||||||
|  | /*************************************************************************** | ||||||
|  |  *   This file is part of awesome-widgets                                  * | ||||||
|  |  *                                                                         * | ||||||
|  |  *   awesome-widgets is free software: you can redistribute it and/or      * | ||||||
|  |  *   modify it under the terms of the GNU General Public License as        * | ||||||
|  |  *   published by the Free Software Foundation, either version 3 of the    * | ||||||
|  |  *   License, or (at your option) any later version.                       * | ||||||
|  |  *                                                                         * | ||||||
|  |  *   awesome-widgets is distributed in the hope that it will be useful,    * | ||||||
|  |  *   but WITHOUT ANY WARRANTY; without even the implied warranty of        * | ||||||
|  |  *   MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the         * | ||||||
|  |  *   GNU General Public License for more details.                          * | ||||||
|  |  *                                                                         * | ||||||
|  |  *   You should have received a copy of the GNU General Public License     * | ||||||
|  |  *   along with awesome-widgets. If not, see http://www.gnu.org/licenses/  * | ||||||
|  |  ***************************************************************************/ | ||||||
|  |  | ||||||
|  |  | ||||||
|  | #ifndef AWABSTRACTPAIRHELPER_H | ||||||
|  | #define AWABSTRACTPAIRHELPER_H | ||||||
|  |  | ||||||
|  | #include <QHash> | ||||||
|  |  | ||||||
|  |  | ||||||
|  | class AWAbstractPairHelper | ||||||
|  | { | ||||||
|  | public: | ||||||
|  |     explicit AWAbstractPairHelper(QString _filePath = "", QString _section = ""); | ||||||
|  |     virtual ~AWAbstractPairHelper(); | ||||||
|  |     [[nodiscard]] QStringList keys() const; | ||||||
|  |     [[nodiscard]] QHash<QString, QString> pairs() const; | ||||||
|  |     [[nodiscard]] QStringList values() const; | ||||||
|  |     [[nodiscard]] QSet<QString> valuesSet() const; | ||||||
|  |     // read-write methods | ||||||
|  |     virtual void initItems(); | ||||||
|  |     [[nodiscard]] virtual bool writeItems(const QHash<QString, QString> &_configuration) const; | ||||||
|  |     [[nodiscard]] virtual bool removeUnusedKeys(const QStringList &_keys) const; | ||||||
|  |     // configuration related | ||||||
|  |     virtual void editPairs() = 0; | ||||||
|  |     virtual QStringList leftKeys() = 0; | ||||||
|  |     virtual QStringList rightKeys() = 0; | ||||||
|  |  | ||||||
|  | private: | ||||||
|  |     // properties | ||||||
|  |     QHash<QString, QString> m_pairs; | ||||||
|  |     QString m_filePath; | ||||||
|  |     QString m_section; | ||||||
|  | }; | ||||||
|  |  | ||||||
|  |  | ||||||
|  | #endif /* AWABSTRACTPAIRHELPER_H */ | ||||||
							
								
								
									
										75
									
								
								sources/awesome-widget/plugin/awabstractselector.cpp
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										75
									
								
								sources/awesome-widget/plugin/awabstractselector.cpp
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,75 @@ | |||||||
|  | /*************************************************************************** | ||||||
|  |  *   This file is part of awesome-widgets                                  * | ||||||
|  |  *                                                                         * | ||||||
|  |  *   awesome-widgets is free software: you can redistribute it and/or      * | ||||||
|  |  *   modify it under the terms of the GNU General Public License as        * | ||||||
|  |  *   published by the Free Software Foundation, either version 3 of the    * | ||||||
|  |  *   License, or (at your option) any later version.                       * | ||||||
|  |  *                                                                         * | ||||||
|  |  *   awesome-widgets is distributed in the hope that it will be useful,    * | ||||||
|  |  *   but WITHOUT ANY WARRANTY; without even the implied warranty of        * | ||||||
|  |  *   MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the         * | ||||||
|  |  *   GNU General Public License for more details.                          * | ||||||
|  |  *                                                                         * | ||||||
|  |  *   You should have received a copy of the GNU General Public License     * | ||||||
|  |  *   along with awesome-widgets. If not, see http://www.gnu.org/licenses/  * | ||||||
|  |  ***************************************************************************/ | ||||||
|  |  | ||||||
|  | #include "awabstractselector.h" | ||||||
|  | #include "ui_awabstractselector.h" | ||||||
|  |  | ||||||
|  | #include "awdebug.h" | ||||||
|  |  | ||||||
|  |  | ||||||
|  | AWAbstractSelector::AWAbstractSelector(QWidget *_parent, const QPair<bool, bool> &_editable) | ||||||
|  |     : QWidget(_parent) | ||||||
|  |     , ui(new Ui::AWAbstractSelector) | ||||||
|  | { | ||||||
|  |     qCDebug(LOG_AW) << __PRETTY_FUNCTION__; | ||||||
|  |  | ||||||
|  |     ui->setupUi(this); | ||||||
|  |     ui->comboBox_key->setEditable(_editable.first); | ||||||
|  |     ui->comboBox_value->setEditable(_editable.second); | ||||||
|  |  | ||||||
|  |     connect(ui->comboBox_key, SIGNAL(currentIndexChanged(int)), this, SIGNAL(selectionChanged())); | ||||||
|  |     connect(ui->comboBox_value, SIGNAL(currentIndexChanged(int)), this, SIGNAL(selectionChanged())); | ||||||
|  | } | ||||||
|  |  | ||||||
|  |  | ||||||
|  | AWAbstractSelector::~AWAbstractSelector() | ||||||
|  | { | ||||||
|  |     qCDebug(LOG_AW) << __PRETTY_FUNCTION__; | ||||||
|  |  | ||||||
|  |     delete ui; | ||||||
|  | } | ||||||
|  |  | ||||||
|  |  | ||||||
|  | QPair<QString, QString> AWAbstractSelector::current() const | ||||||
|  | { | ||||||
|  |     QString key = ui->comboBox_key->currentText(); | ||||||
|  |     QString value = ui->comboBox_value->currentText(); | ||||||
|  |  | ||||||
|  |     return QPair<QString, QString>(key, value); | ||||||
|  | } | ||||||
|  |  | ||||||
|  |  | ||||||
|  | void AWAbstractSelector::init(const QStringList &_keys, const QStringList &_values, | ||||||
|  |                               const QPair<QString, QString> &_current) | ||||||
|  | { | ||||||
|  |     if ((!_keys.contains(_current.first)) || (!_values.contains(_current.second))) { | ||||||
|  |         qCWarning(LOG_AW) << "Invalid current value" << _current << "not found in default ones"; | ||||||
|  |         return; | ||||||
|  |     } | ||||||
|  |     qCDebug(LOG_AW) << "Init selector with keys" << _keys << "and values" << _values << "and current ones are" | ||||||
|  |                     << _current; | ||||||
|  |  | ||||||
|  |     // set data | ||||||
|  |     ui->comboBox_key->clear(); | ||||||
|  |     ui->comboBox_key->addItems(_keys); | ||||||
|  |     ui->comboBox_value->clear(); | ||||||
|  |     ui->comboBox_value->addItems(_values); | ||||||
|  |  | ||||||
|  |     // set current values | ||||||
|  |     ui->comboBox_key->setCurrentText(_current.first); | ||||||
|  |     ui->comboBox_value->setCurrentText(_current.second); | ||||||
|  | } | ||||||
							
								
								
									
										48
									
								
								sources/awesome-widget/plugin/awabstractselector.h
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										48
									
								
								sources/awesome-widget/plugin/awabstractselector.h
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,48 @@ | |||||||
|  | /*************************************************************************** | ||||||
|  |  *   This file is part of awesome-widgets                                  * | ||||||
|  |  *                                                                         * | ||||||
|  |  *   awesome-widgets is free software: you can redistribute it and/or      * | ||||||
|  |  *   modify it under the terms of the GNU General Public License as        * | ||||||
|  |  *   published by the Free Software Foundation, either version 3 of the    * | ||||||
|  |  *   License, or (at your option) any later version.                       * | ||||||
|  |  *                                                                         * | ||||||
|  |  *   awesome-widgets is distributed in the hope that it will be useful,    * | ||||||
|  |  *   but WITHOUT ANY WARRANTY; without even the implied warranty of        * | ||||||
|  |  *   MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the         * | ||||||
|  |  *   GNU General Public License for more details.                          * | ||||||
|  |  *                                                                         * | ||||||
|  |  *   You should have received a copy of the GNU General Public License     * | ||||||
|  |  *   along with awesome-widgets. If not, see http://www.gnu.org/licenses/  * | ||||||
|  |  ***************************************************************************/ | ||||||
|  |  | ||||||
|  |  | ||||||
|  | #ifndef AWABSTRACTSELECTOR_H | ||||||
|  | #define AWABSTRACTSELECTOR_H | ||||||
|  |  | ||||||
|  | #include <QWidget> | ||||||
|  |  | ||||||
|  |  | ||||||
|  | namespace Ui | ||||||
|  | { | ||||||
|  | class AWAbstractSelector; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | class AWAbstractSelector : public QWidget | ||||||
|  | { | ||||||
|  |     Q_OBJECT | ||||||
|  |  | ||||||
|  | public: | ||||||
|  |     explicit AWAbstractSelector(QWidget *_parent = nullptr, const QPair<bool, bool> &_editable = {false, false}); | ||||||
|  |     ~AWAbstractSelector() override; | ||||||
|  |     [[nodiscard]] QPair<QString, QString> current() const; | ||||||
|  |     void init(const QStringList &_keys, const QStringList &_values, const QPair<QString, QString> &_current); | ||||||
|  |  | ||||||
|  | signals: | ||||||
|  |     void selectionChanged(); | ||||||
|  |  | ||||||
|  | private: | ||||||
|  |     Ui::AWAbstractSelector *ui = nullptr; | ||||||
|  | }; | ||||||
|  |  | ||||||
|  |  | ||||||
|  | #endif /* AWABSTRACTSELECTOR_H */ | ||||||
							
								
								
									
										36
									
								
								sources/awesome-widget/plugin/awabstractselector.ui
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										36
									
								
								sources/awesome-widget/plugin/awabstractselector.ui
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,36 @@ | |||||||
|  | <?xml version="1.0" encoding="UTF-8"?> | ||||||
|  | <ui version="4.0"> | ||||||
|  |  <class>AWAbstractSelector</class> | ||||||
|  |  <widget class="QWidget" name="AWAbstractSelector"> | ||||||
|  |   <property name="geometry"> | ||||||
|  |    <rect> | ||||||
|  |     <x>0</x> | ||||||
|  |     <y>0</y> | ||||||
|  |     <width>400</width> | ||||||
|  |     <height>25</height> | ||||||
|  |    </rect> | ||||||
|  |   </property> | ||||||
|  |   <layout class="QHBoxLayout" name="horizontalLayout"> | ||||||
|  |    <property name="leftMargin"> | ||||||
|  |     <number>0</number> | ||||||
|  |    </property> | ||||||
|  |    <property name="topMargin"> | ||||||
|  |     <number>0</number> | ||||||
|  |    </property> | ||||||
|  |    <property name="rightMargin"> | ||||||
|  |     <number>0</number> | ||||||
|  |    </property> | ||||||
|  |    <property name="bottomMargin"> | ||||||
|  |     <number>0</number> | ||||||
|  |    </property> | ||||||
|  |    <item> | ||||||
|  |     <widget class="QComboBox" name="comboBox_key"/> | ||||||
|  |    </item> | ||||||
|  |    <item> | ||||||
|  |     <widget class="QComboBox" name="comboBox_value"/> | ||||||
|  |    </item> | ||||||
|  |   </layout> | ||||||
|  |  </widget> | ||||||
|  |  <resources/> | ||||||
|  |  <connections/> | ||||||
|  | </ui> | ||||||
| @ -21,26 +21,22 @@ | |||||||
| #include <KNotifications/KNotification> | #include <KNotifications/KNotification> | ||||||
|  |  | ||||||
| #include <QDesktopServices> | #include <QDesktopServices> | ||||||
| #include <QJsonDocument> | #include <QFile> | ||||||
| #include <QJsonParseError> |  | ||||||
| #include <QMessageBox> |  | ||||||
| #include <QNetworkAccessManager> |  | ||||||
| #include <QNetworkRequest> |  | ||||||
| #include <QNetworkReply> |  | ||||||
| #include <QProcess> | #include <QProcess> | ||||||
| #include <QProcessEnvironment> | #include <QUrl> | ||||||
| #include <QStandardPaths> |  | ||||||
|  |  | ||||||
| #include <fontdialog/fontdialog.h> | #include <fontdialog/fontdialog.h> | ||||||
|  |  | ||||||
| #include "awdebug.h" | #include "awdebug.h" | ||||||
| #include "version.h" | #include "awupdatehelper.h" | ||||||
|  |  | ||||||
|  |  | ||||||
| AWActions::AWActions(QObject *parent) | AWActions::AWActions(QObject *_parent) | ||||||
|     : QObject(parent) |     : QObject(_parent) | ||||||
| { | { | ||||||
|     qCDebug(LOG_AW) << __PRETTY_FUNCTION__; |     qCDebug(LOG_AW) << __PRETTY_FUNCTION__; | ||||||
|  |  | ||||||
|  |     m_updateHelper = new AWUpdateHelper(this); | ||||||
| } | } | ||||||
|  |  | ||||||
|  |  | ||||||
| @ -50,208 +46,103 @@ AWActions::~AWActions() | |||||||
| } | } | ||||||
|  |  | ||||||
|  |  | ||||||
| void AWActions::checkUpdates(const bool showAnyway) | void AWActions::checkUpdates(const bool _showAnyway) | ||||||
| { | { | ||||||
|     qCDebug(LOG_AW) << "Show anyway" << showAnyway; |     qCDebug(LOG_AW) << "Show anyway" << _showAnyway; | ||||||
|  |  | ||||||
|     // showAnyway options requires to show message if no updates found on direct |     if (!m_updateHelper->checkVersion()) | ||||||
|     // request. In case of automatic check no message will be shown |         m_updateHelper->checkUpdates(_showAnyway); | ||||||
|     QNetworkAccessManager *manager = new QNetworkAccessManager(nullptr); | } | ||||||
|     connect(manager, &QNetworkAccessManager::finished, |  | ||||||
|             [showAnyway, this](QNetworkReply *reply) { |  | ||||||
|                 return versionReplyRecieved(reply, showAnyway); |  | ||||||
|             }); |  | ||||||
|  |  | ||||||
|     manager->get(QNetworkRequest(QUrl(VERSION_API))); |  | ||||||
|  | QString AWActions::getFileContent(const QString &_path) | ||||||
|  | { | ||||||
|  |     qCDebug(LOG_AW) << "Get content from file" << _path; | ||||||
|  |  | ||||||
|  |     QFile inputFile(_path); | ||||||
|  |     if (!inputFile.open(QIODevice::ReadOnly | QIODevice::Text)) { | ||||||
|  |         qCWarning(LOG_AW) << "Could not open file as text" << inputFile.fileName(); | ||||||
|  |         return ""; | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     QString output = inputFile.readAll(); | ||||||
|  |     inputFile.close(); | ||||||
|  |     return output; | ||||||
| } | } | ||||||
|  |  | ||||||
|  |  | ||||||
| // HACK: since QML could not use QLoggingCategory I need this hack | // HACK: since QML could not use QLoggingCategory I need this hack | ||||||
| bool AWActions::isDebugEnabled() const | bool AWActions::isDebugEnabled() | ||||||
| { | { | ||||||
|     return LOG_AW().isDebugEnabled(); |     return LOG_AW().isDebugEnabled(); | ||||||
| } | } | ||||||
|  |  | ||||||
|  |  | ||||||
| bool AWActions::runCmd(const QString cmd) const | bool AWActions::runCmd(const QString &_cmd, const QStringList &_args) | ||||||
| { | { | ||||||
|     qCDebug(LOG_AW) << "Cmd" << cmd; |     qCDebug(LOG_AW) << "Cmd" << _cmd << "args" << _args; | ||||||
|  |  | ||||||
|     sendNotification(QString("Info"), i18n("Run %1", cmd)); |     sendNotification(QString("Info"), i18n("Run %1", _cmd)); | ||||||
|  |  | ||||||
|     return QProcess::startDetached(cmd); |     return QProcess::startDetached(_cmd, _args); | ||||||
| } | } | ||||||
|  |  | ||||||
|  |  | ||||||
| // HACK: this method uses variable from version.h | // HACK: this method uses variable from version.h | ||||||
| void AWActions::showReadme() const | void AWActions::showReadme() | ||||||
| { | { | ||||||
|     QDesktopServices::openUrl(QString(HOMEPAGE)); |     QDesktopServices::openUrl(QUrl(HOMEPAGE)); | ||||||
|  | } | ||||||
|  |  | ||||||
|  |  | ||||||
|  | void AWActions::showLegacyInfo() | ||||||
|  | { | ||||||
|  |     auto *msgBox = new QMessageBox(nullptr); | ||||||
|  |     msgBox->setAttribute(Qt::WA_DeleteOnClose); | ||||||
|  |     msgBox->setModal(false); | ||||||
|  |     msgBox->setWindowTitle(i18n("Not supported")); | ||||||
|  |     msgBox->setText(i18n("You are using mammoth's Qt version, try to update it first")); | ||||||
|  |     msgBox->setStandardButtons(QMessageBox::Ok); | ||||||
|  |     msgBox->setIcon(QMessageBox::Information); | ||||||
|  |  | ||||||
|  |     msgBox->open(); | ||||||
| } | } | ||||||
|  |  | ||||||
|  |  | ||||||
| // HACK: this method uses variables from version.h | // HACK: this method uses variables from version.h | ||||||
| QString AWActions::getAboutText(const QString type) const | QString AWActions::getAboutText(const QString &_type) | ||||||
| { | { | ||||||
|     qCDebug(LOG_AW) << "Type" << type; |     qCDebug(LOG_AW) << "Type" << _type; | ||||||
|  |  | ||||||
|     QString text; |     return AWDebug::getAboutText(_type); | ||||||
|     if (type == QString("header")) { |  | ||||||
|         text = QString(NAME); |  | ||||||
|     } else if (type == QString("version")) { |  | ||||||
|         text = i18n("Version %1 (build date %2)", QString(VERSION), |  | ||||||
|                     QString(BUILD_DATE)); |  | ||||||
|         if (!QString(COMMIT_SHA).isEmpty()) |  | ||||||
|             text += QString(" (%1)").arg(QString(COMMIT_SHA)); |  | ||||||
|     } else if (type == QString("description")) { |  | ||||||
|         text = i18n("A set of minimalistic plasmoid widgets"); |  | ||||||
|     } else if (type == QString("links")) { |  | ||||||
|         text = i18n("Links:") + QString("<br>") |  | ||||||
|                + QString("<a href=\"%1\">%2</a><br>") |  | ||||||
|                      .arg(QString(HOMEPAGE)) |  | ||||||
|                      .arg(i18n("Homepage")) |  | ||||||
|                + QString("<a href=\"%1\">%2</a><br>") |  | ||||||
|                      .arg(QString(REPOSITORY)) |  | ||||||
|                      .arg(i18n("Repository")) |  | ||||||
|                + QString("<a href=\"%1\">%2</a><br>") |  | ||||||
|                      .arg(QString(BUGTRACKER)) |  | ||||||
|                      .arg(i18n("Bugtracker")) |  | ||||||
|                + QString("<a href=\"%1\">%2</a><br>") |  | ||||||
|                      .arg(QString(TRANSLATION)) |  | ||||||
|                      .arg(i18n("Translation issue")) |  | ||||||
|                + QString("<a href=\"%1\">%2</a><br>") |  | ||||||
|                      .arg(QString(AUR_PACKAGES)) |  | ||||||
|                      .arg(i18n("AUR packages")) |  | ||||||
|                + QString("<a href=\"%1\">%2</a>") |  | ||||||
|                      .arg(QString(OPENSUSE_PACKAGES)) |  | ||||||
|                      .arg(i18n("openSUSE packages")); |  | ||||||
|     } else if (type == QString("copy")) { |  | ||||||
|         text = QString("<small>© %1 <a href=\"mailto:%2\">%3</a><br>") |  | ||||||
|                    .arg(QString(DATE)) |  | ||||||
|                    .arg(QString(EMAIL)) |  | ||||||
|                    .arg(QString(AUTHOR)) |  | ||||||
|                + i18n("This software is licensed under %1", QString(LICENSE)) |  | ||||||
|                + QString("</small>"); |  | ||||||
|     } else if (type == QString("translators")) { |  | ||||||
|         text = i18n("Translators: %1", QString(TRANSLATORS)); |  | ||||||
|     } else if (type == QString("3rdparty")) { |  | ||||||
|         QStringList trdPartyList |  | ||||||
|             = QString(TRDPARTY_LICENSE) |  | ||||||
|                   .split(QChar(';'), QString::SkipEmptyParts); |  | ||||||
|         for (int i = 0; i < trdPartyList.count(); i++) |  | ||||||
|             trdPartyList[i] = QString("<a href=\"%3\">%1</a> (%2 license)") |  | ||||||
|                                   .arg(trdPartyList.at(i).split(QChar(','))[0]) |  | ||||||
|                                   .arg(trdPartyList.at(i).split(QChar(','))[1]) |  | ||||||
|                                   .arg(trdPartyList.at(i).split(QChar(','))[2]); |  | ||||||
|         text = i18n("This software uses: %1", trdPartyList.join(QString(", "))); |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     return text; |  | ||||||
| } | } | ||||||
|  |  | ||||||
|  |  | ||||||
| QVariantMap AWActions::getFont(const QVariantMap defaultFont) const | QVariantMap AWActions::getFont(const QVariantMap &_defaultFont) | ||||||
| { | { | ||||||
|     qCDebug(LOG_AW) << "Default font is" << defaultFont; |     qCDebug(LOG_AW) << "Default font is" << _defaultFont; | ||||||
|  |  | ||||||
|     QVariantMap fontMap; |     QVariantMap fontMap; | ||||||
|     CFont defaultCFont = CFont(defaultFont[QString("family")].toString(), |     int ret = 0; | ||||||
|                                defaultFont[QString("size")].toInt(), 400, false, |     CFont defaultCFont = CFont(_defaultFont["family"].toString(), _defaultFont["size"].toInt(), 400, false, | ||||||
|                                defaultFont[QString("color")].toString()); |                                _defaultFont["color"].toString()); | ||||||
|     CFont font |     CFont font = CFontDialog::getFont(i18n("Select font"), defaultCFont, false, false, &ret); | ||||||
|         = CFontDialog::getFont(i18n("Select font"), defaultCFont, false, false); |  | ||||||
|     fontMap[QString("color")] = font.color().name(); |     fontMap["applied"] = ret; | ||||||
|     fontMap[QString("family")] = font.family(); |     fontMap["color"] = font.color().name(); | ||||||
|     fontMap[QString("size")] = font.pointSize(); |     fontMap["family"] = font.family(); | ||||||
|  |     fontMap["size"] = font.pointSize(); | ||||||
|  |  | ||||||
|     return fontMap; |     return fontMap; | ||||||
| } | } | ||||||
|  |  | ||||||
|  |  | ||||||
| // to avoid additional object definition this method is static | // to avoid additional object definition this method is static | ||||||
| void AWActions::sendNotification(const QString eventId, const QString message) | void AWActions::sendNotification(const QString &_eventId, const QString &_message) | ||||||
| { | { | ||||||
|     qCDebug(LOG_AW) << "Event" << eventId; |     qCDebug(LOG_AW) << "Event" << _eventId << "with message" << _message; | ||||||
|     qCDebug(LOG_AW) << "Message" << message; |  | ||||||
|  |  | ||||||
|     KNotification *notification = KNotification::event( |     KNotification *notification | ||||||
|         eventId, QString("Awesome Widget ::: %1").arg(eventId), message); |         = KNotification::event(_eventId, QString("Awesome Widget ::: %1").arg(_eventId), _message); | ||||||
|     notification->setComponentName( |     notification->setComponentName("plasma-applet-org.kde.plasma.awesome-widget"); | ||||||
|         QString("plasma-applet-org.kde.plasma.awesome-widget")); |  | ||||||
| } |  | ||||||
|  |  | ||||||
|  |  | ||||||
| void AWActions::showInfo(const QString version) const |  | ||||||
| { |  | ||||||
|     qCDebug(LOG_AW) << "Version" << version; |  | ||||||
|  |  | ||||||
|     QString text = i18n("You are using the actual version %1", version); |  | ||||||
|     if (!QString(COMMIT_SHA).isEmpty()) |  | ||||||
|         text += QString(" (%1)").arg(QString(COMMIT_SHA)); |  | ||||||
|     QMessageBox::information(nullptr, i18n("No new version found"), text); |  | ||||||
| } |  | ||||||
|  |  | ||||||
|  |  | ||||||
| void AWActions::showUpdates(const QString version) const |  | ||||||
| { |  | ||||||
|     qCDebug(LOG_AW) << "Version" << version; |  | ||||||
|  |  | ||||||
|     QString text; |  | ||||||
|     text += i18n("Current version : %1", QString(VERSION)); |  | ||||||
|     text += QString(COMMIT_SHA).isEmpty() |  | ||||||
|                 ? QString("\n") |  | ||||||
|                 : QString(" (%1)\n").arg(QString(COMMIT_SHA)); |  | ||||||
|     text += i18n("New version : %1", version) + QString("\n\n"); |  | ||||||
|     text += i18n("Click \"Ok\" to download"); |  | ||||||
|  |  | ||||||
|     int select |  | ||||||
|         = QMessageBox::information(nullptr, i18n("There are updates"), text, |  | ||||||
|                                    QMessageBox::Ok | QMessageBox::Cancel); |  | ||||||
|     switch (select) { |  | ||||||
|     case QMessageBox::Ok: |  | ||||||
|         QDesktopServices::openUrl(QString(RELEASES) + version); |  | ||||||
|         break; |  | ||||||
|     case QMessageBox::Cancel: |  | ||||||
|     default: |  | ||||||
|         break; |  | ||||||
|     } |  | ||||||
| } |  | ||||||
|  |  | ||||||
|  |  | ||||||
| void AWActions::versionReplyRecieved(QNetworkReply *reply, |  | ||||||
|                                      const bool showAnyway) const |  | ||||||
| { |  | ||||||
|     qCDebug(LOG_AW) << "Return code" << reply->error(); |  | ||||||
|     qCDebug(LOG_AW) << "Reply error message" << reply->errorString(); |  | ||||||
|     qCDebug(LOG_AW) << "Show anyway" << showAnyway; |  | ||||||
|  |  | ||||||
|     QJsonParseError error; |  | ||||||
|     QJsonDocument jsonDoc = QJsonDocument::fromJson(reply->readAll(), &error); |  | ||||||
|     reply->deleteLater(); |  | ||||||
|     if ((reply->error() != QNetworkReply::NoError) |  | ||||||
|         || (error.error != QJsonParseError::NoError)) { |  | ||||||
|         qCWarning(LOG_AW) << "Parse error" << error.errorString(); |  | ||||||
|         return; |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     // convert to map |  | ||||||
|     QVariantMap firstRelease = jsonDoc.toVariant().toList().first().toMap(); |  | ||||||
|     QString version = firstRelease[QString("tag_name")].toString(); |  | ||||||
|     version.remove(QString("V.")); |  | ||||||
|     qCInfo(LOG_AW) << "Found version" << version; |  | ||||||
|  |  | ||||||
|     // FIXME: possible there is a better way to check versions |  | ||||||
|     int old_major = QString(VERSION).split(QChar('.')).at(0).toInt(); |  | ||||||
|     int old_minor = QString(VERSION).split(QChar('.')).at(1).toInt(); |  | ||||||
|     int old_patch = QString(VERSION).split(QChar('.')).at(2).toInt(); |  | ||||||
|     int new_major = QString(version).split(QChar('.')).at(0).toInt(); |  | ||||||
|     int new_minor = QString(version).split(QChar('.')).at(1).toInt(); |  | ||||||
|     int new_patch = QString(version).split(QChar('.')).at(2).toInt(); |  | ||||||
|     if ((old_major < new_major) |  | ||||||
|         || ((old_major == new_major) && (old_minor < new_minor)) |  | ||||||
|         || ((old_major == new_major) && (old_minor == new_minor) |  | ||||||
|             && (old_patch < new_patch))) |  | ||||||
|         return showUpdates(version); |  | ||||||
|     else if (showAnyway) |  | ||||||
|         return showInfo(version); |  | ||||||
| } | } | ||||||
|  | |||||||
| @ -19,36 +19,34 @@ | |||||||
| #ifndef AWACTIONS_H | #ifndef AWACTIONS_H | ||||||
| #define AWACTIONS_H | #define AWACTIONS_H | ||||||
|  |  | ||||||
|  | #include <QMap> | ||||||
| #include <QObject> | #include <QObject> | ||||||
|  |  | ||||||
|  |  | ||||||
| class QNetworkReply; | class AWUpdateHelper; | ||||||
|  |  | ||||||
| class AWActions : public QObject | class AWActions : public QObject | ||||||
| { | { | ||||||
|     Q_OBJECT |     Q_OBJECT | ||||||
|  |  | ||||||
| public: | public: | ||||||
|     explicit AWActions(QObject *parent = nullptr); |     explicit AWActions(QObject *_parent = nullptr); | ||||||
|     virtual ~AWActions(); |     ~AWActions() override; | ||||||
|     Q_INVOKABLE void checkUpdates(const bool showAnyway = false); |     Q_INVOKABLE void checkUpdates(bool _showAnyway = false); | ||||||
|     Q_INVOKABLE bool isDebugEnabled() const; |     Q_INVOKABLE static QString getFileContent(const QString &_path); | ||||||
|     Q_INVOKABLE bool runCmd(const QString cmd = QString("/usr/bin/true")) const; |     Q_INVOKABLE static bool isDebugEnabled(); | ||||||
|     Q_INVOKABLE void showReadme() const; |     Q_INVOKABLE static bool runCmd(const QString &_cmd, const QStringList &_args); | ||||||
|  |     Q_INVOKABLE static void showLegacyInfo(); | ||||||
|  |     Q_INVOKABLE static void showReadme(); | ||||||
|     // configuration slots |     // configuration slots | ||||||
|     Q_INVOKABLE QString getAboutText(const QString type |     Q_INVOKABLE static QString getAboutText(const QString &_type); | ||||||
|                                      = QString("header")) const; |     Q_INVOKABLE static QVariantMap getFont(const QVariantMap &_defaultFont); | ||||||
|     Q_INVOKABLE QVariantMap getFont(const QVariantMap defaultFont) const; |  | ||||||
|  |  | ||||||
| public slots: | public slots: | ||||||
|     Q_INVOKABLE static void sendNotification(const QString eventId, |     Q_INVOKABLE static void sendNotification(const QString &_eventId, const QString &_message); | ||||||
|                                              const QString message); |  | ||||||
|  |  | ||||||
| private slots: | private: | ||||||
|     void showInfo(const QString version) const; |     AWUpdateHelper *m_updateHelper = nullptr; | ||||||
|     void showUpdates(const QString version) const; |  | ||||||
|     void versionReplyRecieved(QNetworkReply *reply, |  | ||||||
|                               const bool showAnyway) const; |  | ||||||
| }; | }; | ||||||
|  |  | ||||||
|  |  | ||||||
|  | |||||||
							
								
								
									
										149
									
								
								sources/awesome-widget/plugin/awbugreporter.cpp
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										149
									
								
								sources/awesome-widget/plugin/awbugreporter.cpp
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,149 @@ | |||||||
|  | /*************************************************************************** | ||||||
|  |  *   This file is part of awesome-widgets                                  * | ||||||
|  |  *                                                                         * | ||||||
|  |  *   awesome-widgets is free software: you can redistribute it and/or      * | ||||||
|  |  *   modify it under the terms of the GNU General Public License as        * | ||||||
|  |  *   published by the Free Software Foundation, either version 3 of the    * | ||||||
|  |  *   License, or (at your option) any later version.                       * | ||||||
|  |  *                                                                         * | ||||||
|  |  *   awesome-widgets is distributed in the hope that it will be useful,    * | ||||||
|  |  *   but WITHOUT ANY WARRANTY; without even the implied warranty of        * | ||||||
|  |  *   MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the         * | ||||||
|  |  *   GNU General Public License for more details.                          * | ||||||
|  |  *                                                                         * | ||||||
|  |  *   You should have received a copy of the GNU General Public License     * | ||||||
|  |  *   along with awesome-widgets. If not, see http://www.gnu.org/licenses/  * | ||||||
|  |  ***************************************************************************/ | ||||||
|  |  | ||||||
|  | #include "awbugreporter.h" | ||||||
|  |  | ||||||
|  | #include <KI18n/KLocalizedString> | ||||||
|  |  | ||||||
|  | #include <QDesktopServices> | ||||||
|  | #include <QJsonDocument> | ||||||
|  | #include <QMessageBox> | ||||||
|  | #include <QNetworkAccessManager> | ||||||
|  | #include <QNetworkReply> | ||||||
|  |  | ||||||
|  | #include "awdebug.h" | ||||||
|  |  | ||||||
|  |  | ||||||
|  | AWBugReporter::AWBugReporter(QObject *_parent) | ||||||
|  |     : QObject(_parent) | ||||||
|  | { | ||||||
|  |     qCDebug(LOG_AW) << __PRETTY_FUNCTION__; | ||||||
|  | } | ||||||
|  |  | ||||||
|  |  | ||||||
|  | AWBugReporter::~AWBugReporter() | ||||||
|  | { | ||||||
|  |     qCDebug(LOG_AW) << __PRETTY_FUNCTION__; | ||||||
|  | } | ||||||
|  |  | ||||||
|  |  | ||||||
|  | void AWBugReporter::doConnect() | ||||||
|  | { | ||||||
|  |     // additional method for testing needs | ||||||
|  |     connect(this, SIGNAL(replyReceived(const int, const QString &)), this, SLOT(showInformation(int, const QString &))); | ||||||
|  | } | ||||||
|  |  | ||||||
|  |  | ||||||
|  | QString AWBugReporter::generateText(const QString &_description, const QString &_reproduce, const QString &_expected, | ||||||
|  |                                     const QString &_logs) | ||||||
|  | { | ||||||
|  |     // do not log _logs here, it may have quite large size | ||||||
|  |     qCDebug(LOG_AW) << "Generate text with description" << _description << "steps" << _reproduce | ||||||
|  |                     << "and expected result" << _expected; | ||||||
|  |  | ||||||
|  |     QString output; | ||||||
|  |     output += QString("**Description**\n\n%1\n\n").arg(_description); | ||||||
|  |     output += QString("**Step to _reproduce**\n\n%1\n\n").arg(_reproduce); | ||||||
|  |     output += QString("**Expected result**\n\n%1\n\n").arg(_expected); | ||||||
|  |     output += QString("**Version**\n\n%1\n\n").arg(AWDebug::getBuildData().join(QString("\n"))); | ||||||
|  |     // append _logs | ||||||
|  |     output += QString("**Logs**\n\n%1").arg(_logs); | ||||||
|  |  | ||||||
|  |     return output; | ||||||
|  | } | ||||||
|  |  | ||||||
|  |  | ||||||
|  | void AWBugReporter::sendBugReport(const QString &_title, const QString &_body) | ||||||
|  | { | ||||||
|  |     qCDebug(LOG_AW) << "Send bug report with title" << _title << "and body" << _body; | ||||||
|  |  | ||||||
|  |     auto *manager = new QNetworkAccessManager(nullptr); | ||||||
|  |     connect(manager, SIGNAL(finished(QNetworkReply *)), this, SLOT(issueReplyRecieved(QNetworkReply *))); | ||||||
|  |  | ||||||
|  |     QNetworkRequest request = QNetworkRequest(QUrl(BUGTRACKER_API)); | ||||||
|  |     request.setHeader(QNetworkRequest::ContentTypeHeader, "application/json"); | ||||||
|  |  | ||||||
|  |     // generate payload | ||||||
|  |     QVariantMap payload; | ||||||
|  |     payload["title"] = _title; | ||||||
|  |     payload["body"] = _body; | ||||||
|  |     payload["labels"] = QStringList() << "from application"; | ||||||
|  |     // convert to QByteArray to send request | ||||||
|  |     QByteArray data = QJsonDocument::fromVariant(payload).toJson(QJsonDocument::Compact); | ||||||
|  |     qCInfo(LOG_AW) << "Send request with _body" << data.data() << "and size" << data.size(); | ||||||
|  |  | ||||||
|  |     manager->post(request, data); | ||||||
|  | } | ||||||
|  |  | ||||||
|  |  | ||||||
|  | void AWBugReporter::issueReplyRecieved(QNetworkReply *_reply) | ||||||
|  | { | ||||||
|  |     if (_reply->error() != QNetworkReply::NoError) { | ||||||
|  |         qCWarning(LOG_AW) << "An error occurs" << _reply->error() << "with message" << _reply->errorString(); | ||||||
|  |         return emit(replyReceived(0, "")); | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     QJsonParseError error{}; | ||||||
|  |     QJsonDocument jsonDoc = QJsonDocument::fromJson(_reply->readAll(), &error); | ||||||
|  |     if (error.error != QJsonParseError::NoError) { | ||||||
|  |         qCWarning(LOG_AW) << "Parse error" << error.errorString(); | ||||||
|  |         return emit(replyReceived(0, "")); | ||||||
|  |     } | ||||||
|  |     _reply->deleteLater(); | ||||||
|  |  | ||||||
|  |     // convert to map | ||||||
|  |     QVariantMap response = jsonDoc.toVariant().toMap(); | ||||||
|  |     QString url = response["html_url"].toString(); | ||||||
|  |     int number = response["number"].toInt(); | ||||||
|  |  | ||||||
|  |     return emit(replyReceived(number, url)); | ||||||
|  | } | ||||||
|  |  | ||||||
|  |  | ||||||
|  | void AWBugReporter::showInformation(const int _number, const QString &_url) | ||||||
|  | { | ||||||
|  |     qCDebug(LOG_AW) << "Created issue with number" << _number << "and url" << _url; | ||||||
|  |  | ||||||
|  |     // cache url first | ||||||
|  |     m_lastBugUrl = _url; | ||||||
|  |  | ||||||
|  |     auto *msgBox = new QMessageBox(nullptr); | ||||||
|  |     msgBox->setAttribute(Qt::WA_DeleteOnClose); | ||||||
|  |     msgBox->setModal(false); | ||||||
|  |     msgBox->setWindowTitle(i18n("Issue created")); | ||||||
|  |     msgBox->setText(i18n("Issue %1 has been created", _number)); | ||||||
|  |     msgBox->setStandardButtons(QMessageBox::Open | QMessageBox::Close); | ||||||
|  |     msgBox->setIcon(QMessageBox::Information); | ||||||
|  |  | ||||||
|  |     msgBox->open(this, SLOT(userReplyOnBugReport(QAbstractButton *))); | ||||||
|  | } | ||||||
|  |  | ||||||
|  |  | ||||||
|  | void AWBugReporter::userReplyOnBugReport(QAbstractButton *_button) | ||||||
|  | { | ||||||
|  |     QMessageBox::ButtonRole ret = dynamic_cast<QMessageBox *>(sender())->buttonRole(_button); | ||||||
|  |     qCInfo(LOG_AW) << "User select" << ret; | ||||||
|  |  | ||||||
|  |     switch (ret) { | ||||||
|  |     case QMessageBox::AcceptRole: | ||||||
|  |         QDesktopServices::openUrl(m_lastBugUrl); | ||||||
|  |         break; | ||||||
|  |     case QMessageBox::RejectRole: | ||||||
|  |     default: | ||||||
|  |         break; | ||||||
|  |     } | ||||||
|  | } | ||||||
							
								
								
									
										53
									
								
								sources/awesome-widget/plugin/awbugreporter.h
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										53
									
								
								sources/awesome-widget/plugin/awbugreporter.h
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,53 @@ | |||||||
|  | /*************************************************************************** | ||||||
|  |  *   This file is part of awesome-widgets                                  * | ||||||
|  |  *                                                                         * | ||||||
|  |  *   awesome-widgets is free software: you can redistribute it and/or      * | ||||||
|  |  *   modify it under the terms of the GNU General Public License as        * | ||||||
|  |  *   published by the Free Software Foundation, either version 3 of the    * | ||||||
|  |  *   License, or (at your option) any later version.                       * | ||||||
|  |  *                                                                         * | ||||||
|  |  *   awesome-widgets is distributed in the hope that it will be useful,    * | ||||||
|  |  *   but WITHOUT ANY WARRANTY; without even the implied warranty of        * | ||||||
|  |  *   MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the         * | ||||||
|  |  *   GNU General Public License for more details.                          * | ||||||
|  |  *                                                                         * | ||||||
|  |  *   You should have received a copy of the GNU General Public License     * | ||||||
|  |  *   along with awesome-widgets. If not, see http://www.gnu.org/licenses/  * | ||||||
|  |  ***************************************************************************/ | ||||||
|  |  | ||||||
|  |  | ||||||
|  | #ifndef AWBUGREPORTER_H | ||||||
|  | #define AWBUGREPORTER_H | ||||||
|  |  | ||||||
|  | #include <QObject> | ||||||
|  |  | ||||||
|  |  | ||||||
|  | class QAbstractButton; | ||||||
|  | class QNetworkReply; | ||||||
|  |  | ||||||
|  | class AWBugReporter : public QObject | ||||||
|  | { | ||||||
|  |     Q_OBJECT | ||||||
|  |  | ||||||
|  | public: | ||||||
|  |     explicit AWBugReporter(QObject *_parent = nullptr); | ||||||
|  |     ~AWBugReporter() override; | ||||||
|  |     Q_INVOKABLE void doConnect(); | ||||||
|  |     Q_INVOKABLE static QString generateText(const QString &_description, const QString &_reproduce, | ||||||
|  |                                             const QString &_expected, const QString &_logs); | ||||||
|  |     Q_INVOKABLE void sendBugReport(const QString &_title, const QString &_body); | ||||||
|  |  | ||||||
|  | signals: | ||||||
|  |     void replyReceived(int _number, const QString &_url); | ||||||
|  |  | ||||||
|  | private slots: | ||||||
|  |     void issueReplyRecieved(QNetworkReply *_reply); | ||||||
|  |     void showInformation(int _number, const QString &_url); | ||||||
|  |     void userReplyOnBugReport(QAbstractButton *_button); | ||||||
|  |  | ||||||
|  | private: | ||||||
|  |     QString m_lastBugUrl; | ||||||
|  | }; | ||||||
|  |  | ||||||
|  |  | ||||||
|  | #endif /* AWBUGREPORTER_H */ | ||||||
| @ -17,25 +17,21 @@ | |||||||
|  |  | ||||||
| #include "awconfighelper.h" | #include "awconfighelper.h" | ||||||
|  |  | ||||||
| #include <KI18n/KLocalizedString> |  | ||||||
|  |  | ||||||
| #include <QCheckBox> |  | ||||||
| #include <QDialogButtonBox> |  | ||||||
| #include <QDir> | #include <QDir> | ||||||
| #include <QFileDialog> |  | ||||||
| #include <QMessageBox> |  | ||||||
| #include <QQmlPropertyMap> | #include <QQmlPropertyMap> | ||||||
| #include <QSettings> | #include <QSettings> | ||||||
|  | #include <QStandardPaths> | ||||||
| #include <QTextCodec> | #include <QTextCodec> | ||||||
| #include <QVBoxLayout> |  | ||||||
|  |  | ||||||
| #include "awdebug.h" | #include "awdebug.h" | ||||||
|  |  | ||||||
|  |  | ||||||
| AWConfigHelper::AWConfigHelper(QObject *parent) | AWConfigHelper::AWConfigHelper(QObject *_parent) | ||||||
|     : QObject(parent) |     : QObject(_parent) | ||||||
| { | { | ||||||
|     qCDebug(LOG_AW) << __PRETTY_FUNCTION__; |     qCDebug(LOG_AW) << __PRETTY_FUNCTION__; | ||||||
|  |  | ||||||
|  |     m_baseDir = QString("%1/awesomewidgets").arg(QStandardPaths::writableLocation(QStandardPaths::GenericDataLocation)); | ||||||
| } | } | ||||||
|  |  | ||||||
|  |  | ||||||
| @ -45,30 +41,41 @@ AWConfigHelper::~AWConfigHelper() | |||||||
| } | } | ||||||
|  |  | ||||||
|  |  | ||||||
| bool AWConfigHelper::dropCache() const | QString AWConfigHelper::configurationDirectory() | ||||||
| { | { | ||||||
|     QString fileName = QString("%1/awesomewidgets.ndx") |     // get readable directory | ||||||
|                            .arg(QStandardPaths::writableLocation( |     QString localDir = QString("%1/awesomewidgets/configs") | ||||||
|                                QStandardPaths::GenericCacheLocation)); |                            .arg(QStandardPaths::writableLocation(QStandardPaths::GenericDataLocation)); | ||||||
|  |  | ||||||
|  |     // create directory and copy files from default settings | ||||||
|  |     QDir localDirectory; | ||||||
|  |     if ((!localDirectory.exists(localDir)) && (localDirectory.mkpath(localDir))) { | ||||||
|  |         qCInfo(LOG_AW) << "Created directory" << localDir; | ||||||
|  |         copyConfigs(localDir); | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     return localDir; | ||||||
|  | } | ||||||
|  |  | ||||||
|  |  | ||||||
|  | bool AWConfigHelper::dropCache() | ||||||
|  | { | ||||||
|  |     QString fileName | ||||||
|  |         = QString("%1/awesomewidgets.ndx").arg(QStandardPaths::writableLocation(QStandardPaths::GenericCacheLocation)); | ||||||
|  |  | ||||||
|     return QFile(fileName).remove(); |     return QFile(fileName).remove(); | ||||||
| } | } | ||||||
|  |  | ||||||
|  |  | ||||||
| void AWConfigHelper::exportConfiguration(QObject *nativeConfig) const | bool AWConfigHelper::exportConfiguration(QObject *_nativeConfig, const QString &_fileName) const | ||||||
| { | { | ||||||
|     // get file path and init settings object |     qCDebug(LOG_AW) << "Selected filename" << _fileName; | ||||||
|     QString fileName = QFileDialog::getSaveFileName(nullptr, i18n("Export")); |  | ||||||
|     if (fileName.isEmpty()) |  | ||||||
|         return; |  | ||||||
|     qCInfo(LOG_AW) << "Selected filename" << fileName; |  | ||||||
|     QSettings settings(fileName, QSettings::IniFormat); |  | ||||||
|  |  | ||||||
|  |     QSettings settings(_fileName, QSettings::IniFormat); | ||||||
|     // plasmoid configuration |     // plasmoid configuration | ||||||
|     QQmlPropertyMap *configuration |     const auto *configuration = dynamic_cast<const QQmlPropertyMap *>(_nativeConfig); | ||||||
|         = static_cast<QQmlPropertyMap *>(nativeConfig); |     settings.beginGroup("plasmoid"); | ||||||
|     settings.beginGroup(QString("plasmoid")); |     for (auto &key : configuration->keys()) { | ||||||
|     foreach (QString key, configuration->keys()) { |  | ||||||
|         QVariant value = configuration->value(key); |         QVariant value = configuration->value(key); | ||||||
|         if (!value.isValid()) |         if (!value.isValid()) | ||||||
|             continue; |             continue; | ||||||
| @ -77,85 +84,76 @@ void AWConfigHelper::exportConfiguration(QObject *nativeConfig) const | |||||||
|     settings.endGroup(); |     settings.endGroup(); | ||||||
|  |  | ||||||
|     // extensions |     // extensions | ||||||
|     foreach (QString item, m_dirs) { |     for (auto &item : m_dirs) { | ||||||
|         QStringList items |         QStringList items | ||||||
|             = QDir(QString("%1/%2").arg(m_baseDir).arg(item)) |             = QDir(QString("%1/%2").arg(m_baseDir).arg(item)).entryList(QStringList() << "*.desktop", QDir::Files); | ||||||
|                   .entryList(QStringList() << QString("*.desktop"), |  | ||||||
|                              QDir::Files); |  | ||||||
|         settings.beginGroup(item); |         settings.beginGroup(item); | ||||||
|         foreach (QString it, items) |         for (auto &it : items) | ||||||
|             copyExtensions(it, item, settings, false); |             copyExtensions(it, item, settings, false); | ||||||
|         settings.endGroup(); |         settings.endGroup(); | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     // additional files |     // additional files | ||||||
|     settings.beginGroup(QString("json")); |     settings.beginGroup("json"); | ||||||
|     // script filters |     // script filters | ||||||
|     readFile(settings, QString("filters"), |     readFile(settings, "filters", QString("%1/scripts/awesomewidgets-extscripts-filters.json").arg(m_baseDir)); | ||||||
|              QString("%1/scripts/awesomewidgets-extscripts-filters.json") |  | ||||||
|                  .arg(m_baseDir)); |  | ||||||
|     // weather icon settings |     // weather icon settings | ||||||
|     readFile(settings, QString("weathers"), |     readFile(settings, "weathers", QString("%1/weather/awesomewidgets-extweather-ids.json").arg(m_baseDir)); | ||||||
|              QString("%1/weather/awesomewidgets-extweather-ids.json") |     settings.endGroup(); | ||||||
|                  .arg(m_baseDir)); |  | ||||||
|  |     settings.beginGroup("ini"); | ||||||
|  |     // formatter settings | ||||||
|  |     readFile(settings, "formatters", QString("%1/formatters/formatters.ini").arg(m_baseDir)); | ||||||
|  |     // custom keys settings | ||||||
|  |     readFile(settings, "custom", QString("%1/custom.ini").arg(m_baseDir)); | ||||||
|     settings.endGroup(); |     settings.endGroup(); | ||||||
|  |  | ||||||
|     // sync settings |     // sync settings | ||||||
|     settings.sync(); |     settings.sync(); | ||||||
|     // show additional message |     // show additional message | ||||||
|     switch (settings.status()) { |     return settings.status() == QSettings::NoError; | ||||||
|     case QSettings::NoError: |  | ||||||
|         QMessageBox::information( |  | ||||||
|             nullptr, i18n("Success"), |  | ||||||
|             i18n("Please note that binary files were not copied")); |  | ||||||
|         break; |  | ||||||
|     default: |  | ||||||
|         QMessageBox::critical(nullptr, i18n("Ooops..."), |  | ||||||
|                               i18n("Could not save configuration file")); |  | ||||||
|         break; |  | ||||||
|     } |  | ||||||
| } | } | ||||||
|  |  | ||||||
|  |  | ||||||
| QVariantMap AWConfigHelper::importConfiguration() const | QVariantMap AWConfigHelper::importConfiguration(const QString &_fileName, const bool _importPlasmoid, | ||||||
|  |                                                 const bool _importExtensions, const bool _importAdds) const | ||||||
| { | { | ||||||
|  |     qCDebug(LOG_AW) << "Selected filename" << _fileName; | ||||||
|  |  | ||||||
|     QVariantMap configuration; |     QVariantMap configuration; | ||||||
|     // get file path and init settings object |     QSettings settings(_fileName, QSettings::IniFormat); | ||||||
|     QString fileName = QFileDialog::getOpenFileName(nullptr, i18n("Import")); |  | ||||||
|     if (fileName.isEmpty()) |  | ||||||
|         return configuration; |  | ||||||
|     qCInfo(LOG_AW) << "Selected filename" << fileName; |  | ||||||
|     QSettings settings(fileName, QSettings::IniFormat); |  | ||||||
|     QHash<QString, bool> selection = selectImport(); |  | ||||||
|  |  | ||||||
|     // extensions |     // extensions | ||||||
|     if (selection[QString("extensions")]) { |     if (_importExtensions) { | ||||||
|         foreach (QString item, m_dirs) { |         for (auto &item : m_dirs) { | ||||||
|             settings.beginGroup(item); |             settings.beginGroup(item); | ||||||
|             foreach (QString it, settings.childGroups()) |             for (auto &it : settings.childGroups()) | ||||||
|                 copyExtensions(it, item, settings, true); |                 copyExtensions(it, item, settings, true); | ||||||
|             settings.endGroup(); |             settings.endGroup(); | ||||||
|         } |         } | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     // additional files |     // additional files | ||||||
|     if (selection[QString("adds")]) { |     if (_importAdds) { | ||||||
|         settings.beginGroup(QString("json")); |         settings.beginGroup("json"); | ||||||
|         // script filters |         // script filters | ||||||
|         writeFile(settings, QString("filters"), |         writeFile(settings, "filters", QString("%1/scripts/awesomewidgets-extscripts-filters.json").arg(m_baseDir)); | ||||||
|                   QString("%1/scripts/awesomewidgets-extscripts-filters.json") |  | ||||||
|                       .arg(m_baseDir)); |  | ||||||
|         // weather icon settings |         // weather icon settings | ||||||
|         writeFile(settings, QString("weathers"), |         writeFile(settings, "weathers", QString("%1/weather/awesomewidgets-extweather-ids.json").arg(m_baseDir)); | ||||||
|                   QString("%1/weather/awesomewidgets-extweather-ids.json") |         settings.endGroup(); | ||||||
|                       .arg(m_baseDir)); |  | ||||||
|  |         settings.beginGroup("ini"); | ||||||
|  |         // formatter settings | ||||||
|  |         writeFile(settings, "formatters", QString("%1/formatters/formatters.ini").arg(m_baseDir)); | ||||||
|  |         // custom keys settings | ||||||
|  |         writeFile(settings, "custom", QString("%1/custom.ini").arg(m_baseDir)); | ||||||
|         settings.endGroup(); |         settings.endGroup(); | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     // plasmoid configuration |     // plasmoid configuration | ||||||
|     if (selection[QString("plasmoid")]) { |     if (_importPlasmoid) { | ||||||
|         settings.beginGroup(QString("plasmoid")); |         settings.beginGroup("plasmoid"); | ||||||
|         foreach (QString key, settings.childKeys()) |         for (auto &key : settings.childKeys()) | ||||||
|             configuration[key] = settings.value(key); |             configuration[key] = settings.value(key); | ||||||
|         settings.endGroup(); |         settings.endGroup(); | ||||||
|     } |     } | ||||||
| @ -164,34 +162,23 @@ QVariantMap AWConfigHelper::importConfiguration() const | |||||||
| } | } | ||||||
|  |  | ||||||
|  |  | ||||||
| QVariantMap AWConfigHelper::readDataEngineConfiguration() const | QVariantMap AWConfigHelper::readDataEngineConfiguration() | ||||||
| { | { | ||||||
|     QString fileName |     QString fileName = QStandardPaths::locate(QStandardPaths::ConfigLocation, "plasma-dataengine-extsysmon.conf"); | ||||||
|         = QStandardPaths::locate(QStandardPaths::ConfigLocation, |  | ||||||
|                                  QString("plasma-dataengine-extsysmon.conf")); |  | ||||||
|     qCInfo(LOG_AW) << "Configuration file" << fileName; |     qCInfo(LOG_AW) << "Configuration file" << fileName; | ||||||
|     QSettings settings(fileName, QSettings::IniFormat); |     QSettings settings(fileName, QSettings::IniFormat); | ||||||
|     QVariantMap configuration; |     QVariantMap configuration; | ||||||
|  |  | ||||||
|     settings.beginGroup(QString("Configuration")); |     settings.beginGroup("Configuration"); | ||||||
|     configuration[QString("ACPIPATH")] = settings.value( |     configuration["ACPIPATH"] = settings.value("ACPIPATH", "/sys/class/power_supply/"); | ||||||
|         QString("ACPIPATH"), QString("/sys/class/power_supply/")); |     configuration["GPUDEV"] = settings.value("GPUDEV", "auto"); | ||||||
|     configuration[QString("GPUDEV")] |     configuration["HDDDEV"] = settings.value("HDDDEV", "all"); | ||||||
|         = settings.value(QString("GPUDEV"), QString("auto")); |     configuration["HDDTEMPCMD"] = settings.value("HDDTEMPCMD", "sudo smartctl -a"); | ||||||
|     configuration[QString("HDDDEV")] |     configuration["MPDADDRESS"] = settings.value("MPDADDRESS", "localhost"); | ||||||
|         = settings.value(QString("HDDDEV"), QString("all")); |     configuration["MPDPORT"] = settings.value("MPDPORT", "6600"); | ||||||
|     configuration[QString("HDDTEMPCMD")] |     configuration["MPRIS"] = settings.value("MPRIS", "auto"); | ||||||
|         = settings.value(QString("HDDTEMPCMD"), QString("sudo smartctl -a")); |     configuration["PLAYER"] = settings.value("PLAYER", "mpris"); | ||||||
|     configuration[QString("MPDADDRESS")] |     configuration["PLAYERSYMBOLS"] = settings.value("PLAYERSYMBOLS", "10"); | ||||||
|         = settings.value(QString("MPDADDRESS"), QString("localhost")); |  | ||||||
|     configuration[QString("MPDPORT")] |  | ||||||
|         = settings.value(QString("MPDPORT"), QString("6600")); |  | ||||||
|     configuration[QString("MPRIS")] |  | ||||||
|         = settings.value(QString("MPRIS"), QString("auto")); |  | ||||||
|     configuration[QString("PLAYER")] |  | ||||||
|         = settings.value(QString("PLAYER"), QString("mpris")); |  | ||||||
|     configuration[QString("PLAYERSYMBOLS")] |  | ||||||
|         = settings.value(QString("PLAYERSYMBOLS"), QString("10")); |  | ||||||
|     settings.endGroup(); |     settings.endGroup(); | ||||||
|  |  | ||||||
|     qCInfo(LOG_AW) << "Configuration" << configuration; |     qCInfo(LOG_AW) << "Configuration" << configuration; | ||||||
| @ -200,146 +187,109 @@ QVariantMap AWConfigHelper::readDataEngineConfiguration() const | |||||||
| } | } | ||||||
|  |  | ||||||
|  |  | ||||||
| void AWConfigHelper::writeDataEngineConfiguration( | bool AWConfigHelper::writeDataEngineConfiguration(const QVariantMap &_configuration) | ||||||
|     const QVariantMap configuration) const |  | ||||||
| { | { | ||||||
|     qCDebug(LOG_AW) << "Configuration" << configuration; |     qCDebug(LOG_AW) << "Configuration" << _configuration; | ||||||
|  |  | ||||||
|     QString fileName |     QString fileName = QString("%1/plasma-dataengine-extsysmon.conf") | ||||||
|         = QStandardPaths::writableLocation(QStandardPaths::ConfigLocation) |                            .arg(QStandardPaths::writableLocation(QStandardPaths::ConfigLocation)); | ||||||
|           + QString("/plasma-dataengine-extsysmon.conf"); |  | ||||||
|     QSettings settings(fileName, QSettings::IniFormat); |     QSettings settings(fileName, QSettings::IniFormat); | ||||||
|     qCInfo(LOG_AW) << "Configuration file" << settings.fileName(); |     qCInfo(LOG_AW) << "Configuration file" << settings.fileName(); | ||||||
|  |  | ||||||
|     settings.beginGroup(QString("Configuration")); |     settings.beginGroup("Configuration"); | ||||||
|     settings.setValue(QString("ACPIPATH"), configuration[QString("ACPIPATH")]); |     settings.setValue("ACPIPATH", _configuration["ACPIPATH"]); | ||||||
|     settings.setValue(QString("GPUDEV"), configuration[QString("GPUDEV")]); |     settings.setValue("GPUDEV", _configuration["GPUDEV"]); | ||||||
|     settings.setValue(QString("HDDDEV"), configuration[QString("HDDDEV")]); |     settings.setValue("HDDDEV", _configuration["HDDDEV"]); | ||||||
|     settings.setValue(QString("HDDTEMPCMD"), |     settings.setValue("HDDTEMPCMD", _configuration["HDDTEMPCMD"]); | ||||||
|                       configuration[QString("HDDTEMPCMD")]); |     settings.setValue("MPDADDRESS", _configuration["MPDADDRESS"]); | ||||||
|     settings.setValue(QString("MPDADDRESS"), |     settings.setValue("MPDPORT", _configuration["MPDPORT"]); | ||||||
|                       configuration[QString("MPDADDRESS")]); |     settings.setValue("MPRIS", _configuration["MPRIS"]); | ||||||
|     settings.setValue(QString("MPDPORT"), configuration[QString("MPDPORT")]); |     settings.setValue("PLAYER", _configuration["PLAYER"]); | ||||||
|     settings.setValue(QString("MPRIS"), configuration[QString("MPRIS")]); |     settings.setValue("PLAYERSYMBOLS", _configuration["PLAYERSYMBOLS"]); | ||||||
|     settings.setValue(QString("PLAYER"), configuration[QString("PLAYER")]); |  | ||||||
|     settings.setValue(QString("PLAYERSYMBOLS"), |  | ||||||
|                       configuration[QString("PLAYERSYMBOLS")]); |  | ||||||
|     settings.endGroup(); |     settings.endGroup(); | ||||||
|  |  | ||||||
|     settings.sync(); |     settings.sync(); | ||||||
|  |  | ||||||
|  |     return (settings.status() == QSettings::NoError); | ||||||
| } | } | ||||||
|  |  | ||||||
|  |  | ||||||
| void AWConfigHelper::copyExtensions(const QString item, const QString type, | void AWConfigHelper::copyConfigs(const QString &_localDir) | ||||||
|                                     QSettings &settings, |  | ||||||
|                                     const bool inverse) const |  | ||||||
| { | { | ||||||
|     qCDebug(LOG_AW) << "Extension" << item; |     qCDebug(LOG_AW) << "Local directory" << _localDir; | ||||||
|     qCDebug(LOG_AW) << "Type" << type; |  | ||||||
|     qCDebug(LOG_AW) << "Inverse" << inverse; |  | ||||||
|  |  | ||||||
|     settings.beginGroup(item); |     QStringList dirs = QStandardPaths::locateAll(QStandardPaths::GenericDataLocation, "awesomewidgets/configs", | ||||||
|     QSettings itemSettings( |                                                  QStandardPaths::LocateDirectory); | ||||||
|         QString("%1/%2/%3").arg(m_baseDir).arg(type).arg(item), |     for (auto &dir : dirs) { | ||||||
|         QSettings::IniFormat); |         if (dir == _localDir) | ||||||
|     itemSettings.beginGroup(QString("Desktop Entry")); |             continue; | ||||||
|     if (inverse) |         QStringList files = QDir(dir).entryList(QDir::Files); | ||||||
|         copySettings(settings, itemSettings); |         for (auto &source : files) { | ||||||
|  |             QString destination = QString("%1/%2").arg(_localDir).arg(source); | ||||||
|  |             bool status = QFile::copy(QString("%1/%2").arg(dir).arg(source), destination); | ||||||
|  |             qCInfo(LOG_AW) << "File" << source << "has been copied to" << destination << "with status" << status; | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  | } | ||||||
|  |  | ||||||
|  |  | ||||||
|  | void AWConfigHelper::copyExtensions(const QString &_item, const QString &_type, QSettings &_settings, | ||||||
|  |                                     const bool _inverse) const | ||||||
|  | { | ||||||
|  |     qCDebug(LOG_AW) << "Extension" << _item << "has type" << _type << "inverse copying" << _inverse; | ||||||
|  |  | ||||||
|  |     _settings.beginGroup(_item); | ||||||
|  |     QSettings itemSettings(QString("%1/%2/%3").arg(m_baseDir).arg(_type).arg(_item), QSettings::IniFormat); | ||||||
|  |     itemSettings.beginGroup("Desktop Entry"); | ||||||
|  |     if (_inverse) | ||||||
|  |         copySettings(_settings, itemSettings); | ||||||
|     else |     else | ||||||
|         copySettings(itemSettings, settings); |         copySettings(itemSettings, _settings); | ||||||
|     itemSettings.endGroup(); |     itemSettings.endGroup(); | ||||||
|     settings.endGroup(); |     _settings.endGroup(); | ||||||
|  |  | ||||||
|     if (inverse) |     if (_inverse) | ||||||
|         itemSettings.sync(); |         itemSettings.sync(); | ||||||
| } | } | ||||||
|  |  | ||||||
|  |  | ||||||
| void AWConfigHelper::copySettings(QSettings &from, QSettings &to) const | void AWConfigHelper::copySettings(QSettings &_from, QSettings &_to) | ||||||
| { | { | ||||||
|     foreach (QString key, from.childKeys()) |     for (auto &key : _from.childKeys()) | ||||||
|         to.setValue(key, from.value(key)); |         _to.setValue(key, _from.value(key)); | ||||||
| } | } | ||||||
|  |  | ||||||
|  |  | ||||||
| void AWConfigHelper::readFile(QSettings &settings, const QString key, | void AWConfigHelper::readFile(QSettings &_settings, const QString &_key, const QString &_fileName) | ||||||
|                               const QString fileName) const |  | ||||||
| { | { | ||||||
|     qCDebug(LOG_AW) << "Key" << key; |     qCDebug(LOG_AW) << "Key" << _key << "from file" << _fileName; | ||||||
|     qCDebug(LOG_AW) << "File" << fileName; |  | ||||||
|  |  | ||||||
|     QFile file(fileName); |     QFile file(_fileName); | ||||||
|     if (file.open(QIODevice::ReadOnly | QIODevice::Text)) { |     if (file.open(QIODevice::ReadOnly | QIODevice::Text)) { | ||||||
|         QString text = QString::fromUtf8(file.readAll()); |         QString text = QString::fromUtf8(file.readAll()); | ||||||
|         file.close(); |         file.close(); | ||||||
|         settings.setValue(key, text); |         _settings.setValue(_key, text); | ||||||
|     } else { |     } else { | ||||||
|         qCWarning(LOG_LIB) << "Could not open" << file.fileName(); |         qCWarning(LOG_AW) << "Could not open to read" << file.fileName(); | ||||||
|     } |     } | ||||||
| } | } | ||||||
|  |  | ||||||
|  |  | ||||||
| QHash<QString, bool> AWConfigHelper::selectImport() const | void AWConfigHelper::writeFile(QSettings &_settings, const QString &_key, const QString &_fileName) | ||||||
| { | { | ||||||
|     QDialog *dialog = new QDialog(nullptr); |     qCDebug(LOG_AW) << "Key" << _key << "to file" << _fileName; | ||||||
|     QCheckBox *importPlasmoidSettings |  | ||||||
|         = new QCheckBox(i18n("Import plasmoid settings"), dialog); |  | ||||||
|     importPlasmoidSettings->setChecked(true); |  | ||||||
|     QCheckBox *importExtensionsSettings |  | ||||||
|         = new QCheckBox(i18n("Import extensions"), dialog); |  | ||||||
|     importExtensionsSettings->setChecked(true); |  | ||||||
|     QCheckBox *importAddsSettings |  | ||||||
|         = new QCheckBox(i18n("Import additional files"), dialog); |  | ||||||
|     importAddsSettings->setChecked(true); |  | ||||||
|     QDialogButtonBox *dialogButtons |  | ||||||
|         = new QDialogButtonBox(QDialogButtonBox::Ok | QDialogButtonBox::Cancel, |  | ||||||
|                                Qt::Horizontal, dialog); |  | ||||||
|     QVBoxLayout *layout = new QVBoxLayout(dialog); |  | ||||||
|     layout->addWidget(importPlasmoidSettings); |  | ||||||
|     layout->addWidget(importExtensionsSettings); |  | ||||||
|     layout->addWidget(importAddsSettings); |  | ||||||
|     layout->addWidget(dialogButtons); |  | ||||||
|     connect(dialogButtons, SIGNAL(accepted()), dialog, SLOT(accept())); |  | ||||||
|     connect(dialogButtons, SIGNAL(rejected()), dialog, SLOT(reject())); |  | ||||||
|  |  | ||||||
|     // get parameters |     if (!_settings.contains(_key)) | ||||||
|     QHash<QString, bool> import; |  | ||||||
|     import[QString("plasmoid")] = false; |  | ||||||
|     import[QString("extensions")] = false; |  | ||||||
|     import[QString("adds")] = false; |  | ||||||
|     switch (dialog->exec()) { |  | ||||||
|     case QDialog::Accepted: |  | ||||||
|         import[QString("plasmoid")] = importPlasmoidSettings->isChecked(); |  | ||||||
|         import[QString("extensions")] = importExtensionsSettings->isChecked(); |  | ||||||
|         import[QString("adds")] = importAddsSettings->isChecked(); |  | ||||||
|         break; |  | ||||||
|     case QDialog::Rejected: |  | ||||||
|     default: |  | ||||||
|         break; |  | ||||||
|     } |  | ||||||
|     dialog->deleteLater(); |  | ||||||
|  |  | ||||||
|     return import; |  | ||||||
| } |  | ||||||
|  |  | ||||||
|  |  | ||||||
| void AWConfigHelper::writeFile(QSettings &settings, const QString key, |  | ||||||
|                                const QString fileName) const |  | ||||||
| { |  | ||||||
|     qCDebug(LOG_AW) << "Key" << key; |  | ||||||
|     qCDebug(LOG_AW) << "File" << fileName; |  | ||||||
|  |  | ||||||
|     if (!settings.contains(key)) |  | ||||||
|         return; |         return; | ||||||
|  |  | ||||||
|     QFile file(fileName); |     QFile file(_fileName); | ||||||
|     if (file.open(QIODevice::WriteOnly | QIODevice::Text)) { |     if (file.open(QIODevice::WriteOnly | QIODevice::Text)) { | ||||||
|         QTextStream out(&file); |         QTextStream out(&file); | ||||||
|         out.setCodec("UTF-8"); |         out.setCodec("UTF-8"); | ||||||
|         out << settings.value(key).toString().toUtf8(); |         out << _settings.value(_key).toString().toUtf8(); | ||||||
|         out.flush(); |         out.flush(); | ||||||
|         file.close(); |         file.close(); | ||||||
|     } else { |     } else { | ||||||
|         qCWarning(LOG_LIB) << "Could not open" << file.fileName(); |         qCWarning(LOG_AW) << "Could not open to write" << file.fileName(); | ||||||
|     } |     } | ||||||
| } | } | ||||||
|  | |||||||
| @ -20,7 +20,7 @@ | |||||||
| #define AWCONFIGHELPER_H | #define AWCONFIGHELPER_H | ||||||
|  |  | ||||||
| #include <QObject> | #include <QObject> | ||||||
| #include <QStandardPaths> | #include <QVariant> | ||||||
|  |  | ||||||
|  |  | ||||||
| class QSettings; | class QSettings; | ||||||
| @ -30,34 +30,27 @@ class AWConfigHelper : public QObject | |||||||
|     Q_OBJECT |     Q_OBJECT | ||||||
|  |  | ||||||
| public: | public: | ||||||
|     explicit AWConfigHelper(QObject *parent = nullptr); |     explicit AWConfigHelper(QObject *_parent = nullptr); | ||||||
|     virtual ~AWConfigHelper(); |     ~AWConfigHelper() override; | ||||||
|     Q_INVOKABLE bool dropCache() const; |     Q_INVOKABLE [[nodiscard]] static QString configurationDirectory(); | ||||||
|     Q_INVOKABLE void exportConfiguration(QObject *nativeConfig) const; |     Q_INVOKABLE static bool dropCache(); | ||||||
|     Q_INVOKABLE QVariantMap importConfiguration() const; |     Q_INVOKABLE bool exportConfiguration(QObject *_nativeConfig, const QString &_fileName) const; | ||||||
|  |     Q_INVOKABLE [[nodiscard]] QVariantMap importConfiguration(const QString &_fileName, bool _importPlasmoid, | ||||||
|  |                                                               bool _importExtensions, bool _importAdds) const; | ||||||
|     // dataengine |     // dataengine | ||||||
|     Q_INVOKABLE QVariantMap readDataEngineConfiguration() const; |     Q_INVOKABLE static QVariantMap readDataEngineConfiguration(); | ||||||
|     Q_INVOKABLE void |     Q_INVOKABLE static bool writeDataEngineConfiguration(const QVariantMap &_configuration); | ||||||
|     writeDataEngineConfiguration(const QVariantMap configuration) const; |  | ||||||
|  |  | ||||||
| private: | private: | ||||||
|     // methods |     // methods | ||||||
|     void copyExtensions(const QString item, const QString type, |     static void copyConfigs(const QString &_localDir); | ||||||
|                         QSettings &settings, const bool inverse) const; |     void copyExtensions(const QString &_item, const QString &_type, QSettings &_settings, bool _inverse) const; | ||||||
|     void copySettings(QSettings &from, QSettings &to) const; |     static void copySettings(QSettings &_from, QSettings &_to); | ||||||
|     void readFile(QSettings &settings, const QString key, |     static void readFile(QSettings &_settings, const QString &_key, const QString &_fileName); | ||||||
|                   const QString fileName) const; |     static void writeFile(QSettings &_settings, const QString &_key, const QString &_fileName); | ||||||
|     QHash<QString, bool> selectImport() const; |  | ||||||
|     void writeFile(QSettings &settings, const QString key, |  | ||||||
|                    const QString fileName) const; |  | ||||||
|     // properties |     // properties | ||||||
|     QString m_baseDir = QString("%1/awesomewidgets") |     QString m_baseDir; | ||||||
|                             .arg(QStandardPaths::writableLocation( |     QStringList m_dirs = {"desktops", "quotes", "scripts", "upgrade", "weather", "formatters"}; | ||||||
|                                 QStandardPaths::GenericDataLocation)); |  | ||||||
|     QStringList m_dirs = QStringList() |  | ||||||
|                          << QString("desktops") << QString("quotes") |  | ||||||
|                          << QString("scripts") << QString("upgrade") |  | ||||||
|                          << QString("weather"); |  | ||||||
| }; | }; | ||||||
|  |  | ||||||
|  |  | ||||||
|  | |||||||
							
								
								
									
										37
									
								
								sources/awesome-widget/plugin/awcustomkeysconfig.cpp
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										37
									
								
								sources/awesome-widget/plugin/awcustomkeysconfig.cpp
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,37 @@ | |||||||
|  | /*************************************************************************** | ||||||
|  |  *   This file is part of awesome-widgets                                  * | ||||||
|  |  *                                                                         * | ||||||
|  |  *   awesome-widgets is free software: you can redistribute it and/or      * | ||||||
|  |  *   modify it under the terms of the GNU General Public License as        * | ||||||
|  |  *   published by the Free Software Foundation, either version 3 of the    * | ||||||
|  |  *   License, or (at your option) any later version.                       * | ||||||
|  |  *                                                                         * | ||||||
|  |  *   awesome-widgets is distributed in the hope that it will be useful,    * | ||||||
|  |  *   but WITHOUT ANY WARRANTY; without even the implied warranty of        * | ||||||
|  |  *   MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the         * | ||||||
|  |  *   GNU General Public License for more details.                          * | ||||||
|  |  *                                                                         * | ||||||
|  |  *   You should have received a copy of the GNU General Public License     * | ||||||
|  |  *   along with awesome-widgets. If not, see http://www.gnu.org/licenses/  * | ||||||
|  |  ***************************************************************************/ | ||||||
|  |  | ||||||
|  | #include "awcustomkeysconfig.h" | ||||||
|  |  | ||||||
|  | #include "awcustomkeyshelper.h" | ||||||
|  | #include "awdebug.h" | ||||||
|  |  | ||||||
|  |  | ||||||
|  | AWCustomKeysConfig::AWCustomKeysConfig(QWidget *_parent, const QStringList &_keys) | ||||||
|  |     : AWAbstractPairConfig(_parent, false, _keys) | ||||||
|  | { | ||||||
|  |     qCDebug(LOG_AW) << __PRETTY_FUNCTION__; | ||||||
|  |  | ||||||
|  |     setEditable(true, false); | ||||||
|  |     initHelper<AWCustomKeysHelper>(); | ||||||
|  | } | ||||||
|  |  | ||||||
|  |  | ||||||
|  | AWCustomKeysConfig::~AWCustomKeysConfig() | ||||||
|  | { | ||||||
|  |     qCDebug(LOG_AW) << __PRETTY_FUNCTION__; | ||||||
|  | } | ||||||
							
								
								
									
										35
									
								
								sources/awesome-widget/plugin/awcustomkeysconfig.h
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										35
									
								
								sources/awesome-widget/plugin/awcustomkeysconfig.h
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,35 @@ | |||||||
|  | /*************************************************************************** | ||||||
|  |  *   This file is part of awesome-widgets                                  * | ||||||
|  |  *                                                                         * | ||||||
|  |  *   awesome-widgets is free software: you can redistribute it and/or      * | ||||||
|  |  *   modify it under the terms of the GNU General Public License as        * | ||||||
|  |  *   published by the Free Software Foundation, either version 3 of the    * | ||||||
|  |  *   License, or (at your option) any later version.                       * | ||||||
|  |  *                                                                         * | ||||||
|  |  *   awesome-widgets is distributed in the hope that it will be useful,    * | ||||||
|  |  *   but WITHOUT ANY WARRANTY; without even the implied warranty of        * | ||||||
|  |  *   MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the         * | ||||||
|  |  *   GNU General Public License for more details.                          * | ||||||
|  |  *                                                                         * | ||||||
|  |  *   You should have received a copy of the GNU General Public License     * | ||||||
|  |  *   along with awesome-widgets. If not, see http://www.gnu.org/licenses/  * | ||||||
|  |  ***************************************************************************/ | ||||||
|  |  | ||||||
|  |  | ||||||
|  | #ifndef AWCUSTOMKEYSCONFIG_H | ||||||
|  | #define AWCUSTOMKEYSCONFIG_H | ||||||
|  |  | ||||||
|  | #include "awabstractpairconfig.h" | ||||||
|  |  | ||||||
|  |  | ||||||
|  | class AWCustomKeysConfig : public AWAbstractPairConfig | ||||||
|  | { | ||||||
|  |     Q_OBJECT | ||||||
|  |  | ||||||
|  | public: | ||||||
|  |     explicit AWCustomKeysConfig(QWidget *_parent = nullptr, const QStringList &_keys = QStringList()); | ||||||
|  |     ~AWCustomKeysConfig() override; | ||||||
|  | }; | ||||||
|  |  | ||||||
|  |  | ||||||
|  | #endif /* AWCUSTOMKEYSCONFIG_H */ | ||||||
							
								
								
									
										78
									
								
								sources/awesome-widget/plugin/awcustomkeyshelper.cpp
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										78
									
								
								sources/awesome-widget/plugin/awcustomkeyshelper.cpp
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,78 @@ | |||||||
|  | /*************************************************************************** | ||||||
|  |  *   This file is part of awesome-widgets                                  * | ||||||
|  |  *                                                                         * | ||||||
|  |  *   awesome-widgets is free software: you can redistribute it and/or      * | ||||||
|  |  *   modify it under the terms of the GNU General Public License as        * | ||||||
|  |  *   published by the Free Software Foundation, either version 3 of the    * | ||||||
|  |  *   License, or (at your option) any later version.                       * | ||||||
|  |  *                                                                         * | ||||||
|  |  *   awesome-widgets is distributed in the hope that it will be useful,    * | ||||||
|  |  *   but WITHOUT ANY WARRANTY; without even the implied warranty of        * | ||||||
|  |  *   MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the         * | ||||||
|  |  *   GNU General Public License for more details.                          * | ||||||
|  |  *                                                                         * | ||||||
|  |  *   You should have received a copy of the GNU General Public License     * | ||||||
|  |  *   along with awesome-widgets. If not, see http://www.gnu.org/licenses/  * | ||||||
|  |  ***************************************************************************/ | ||||||
|  |  | ||||||
|  | #include "awcustomkeyshelper.h" | ||||||
|  |  | ||||||
|  | #include <QSet> | ||||||
|  |  | ||||||
|  | #include "awdebug.h" | ||||||
|  |  | ||||||
|  |  | ||||||
|  | AWCustomKeysHelper::AWCustomKeysHelper(QObject *_parent) | ||||||
|  |     : QObject(_parent) | ||||||
|  |     , AWAbstractPairHelper("awesomewidgets/custom.ini", "Custom") | ||||||
|  | { | ||||||
|  |     qCDebug(LOG_AW) << __PRETTY_FUNCTION__; | ||||||
|  | } | ||||||
|  |  | ||||||
|  |  | ||||||
|  | AWCustomKeysHelper::~AWCustomKeysHelper() | ||||||
|  | { | ||||||
|  |     qCDebug(LOG_AW) << __PRETTY_FUNCTION__; | ||||||
|  | } | ||||||
|  |  | ||||||
|  |  | ||||||
|  | QString AWCustomKeysHelper::source(const QString &_key) const | ||||||
|  | { | ||||||
|  |     qCDebug(LOG_AW) << "Get source by key" << _key; | ||||||
|  |  | ||||||
|  |     return pairs()[_key]; | ||||||
|  | } | ||||||
|  |  | ||||||
|  |  | ||||||
|  | QStringList AWCustomKeysHelper::sources() const | ||||||
|  | { | ||||||
|  |     return valuesSet().values(); | ||||||
|  | } | ||||||
|  |  | ||||||
|  |  | ||||||
|  | QStringList AWCustomKeysHelper::refinedSources() const | ||||||
|  | { | ||||||
|  |     auto allSources = valuesSet(); | ||||||
|  |     QSet<QString> output; | ||||||
|  |  | ||||||
|  |     while (output != allSources) { | ||||||
|  |         output.clear(); | ||||||
|  |         for (auto &src : allSources) | ||||||
|  |             output.insert(pairs().contains(src) ? source(src) : src); | ||||||
|  |         allSources = output; | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     return output.values(); | ||||||
|  | } | ||||||
|  |  | ||||||
|  |  | ||||||
|  | QStringList AWCustomKeysHelper::leftKeys() | ||||||
|  | { | ||||||
|  |     return keys(); | ||||||
|  | } | ||||||
|  |  | ||||||
|  |  | ||||||
|  | QStringList AWCustomKeysHelper::rightKeys() | ||||||
|  | { | ||||||
|  |     return {}; | ||||||
|  | } | ||||||
							
								
								
									
										47
									
								
								sources/awesome-widget/plugin/awcustomkeyshelper.h
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										47
									
								
								sources/awesome-widget/plugin/awcustomkeyshelper.h
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,47 @@ | |||||||
|  | /*************************************************************************** | ||||||
|  |  *   This file is part of awesome-widgets                                  * | ||||||
|  |  *                                                                         * | ||||||
|  |  *   awesome-widgets is free software: you can redistribute it and/or      * | ||||||
|  |  *   modify it under the terms of the GNU General Public License as        * | ||||||
|  |  *   published by the Free Software Foundation, either version 3 of the    * | ||||||
|  |  *   License, or (at your option) any later version.                       * | ||||||
|  |  *                                                                         * | ||||||
|  |  *   awesome-widgets is distributed in the hope that it will be useful,    * | ||||||
|  |  *   but WITHOUT ANY WARRANTY; without even the implied warranty of        * | ||||||
|  |  *   MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the         * | ||||||
|  |  *   GNU General Public License for more details.                          * | ||||||
|  |  *                                                                         * | ||||||
|  |  *   You should have received a copy of the GNU General Public License     * | ||||||
|  |  *   along with awesome-widgets. If not, see http://www.gnu.org/licenses/  * | ||||||
|  |  ***************************************************************************/ | ||||||
|  |  | ||||||
|  |  | ||||||
|  | #ifndef AWCUSTOMKEYSHELPER_H | ||||||
|  | #define AWCUSTOMKEYSHELPER_H | ||||||
|  |  | ||||||
|  | #include <QObject> | ||||||
|  |  | ||||||
|  | #include "awabstractpairhelper.h" | ||||||
|  |  | ||||||
|  |  | ||||||
|  | class AWCustomKeysHelper : public QObject, public AWAbstractPairHelper | ||||||
|  | { | ||||||
|  |     Q_OBJECT | ||||||
|  |  | ||||||
|  | public: | ||||||
|  |     explicit AWCustomKeysHelper(QObject *_parent = nullptr); | ||||||
|  |     ~AWCustomKeysHelper() override; | ||||||
|  |     // get | ||||||
|  |     [[nodiscard]] QString source(const QString &_key) const; | ||||||
|  |     [[nodiscard]] QStringList sources() const; | ||||||
|  |     [[nodiscard]] QStringList refinedSources() const; | ||||||
|  |     // configuration related | ||||||
|  |     void editPairs() override{}; | ||||||
|  |     QStringList leftKeys() override; | ||||||
|  |     QStringList rightKeys() override; | ||||||
|  |  | ||||||
|  | private: | ||||||
|  | }; | ||||||
|  |  | ||||||
|  |  | ||||||
|  | #endif /* AWCUSTOMKEYSHELPER_H */ | ||||||
| @ -19,28 +19,30 @@ | |||||||
|  |  | ||||||
| #include <KI18n/KLocalizedString> | #include <KI18n/KLocalizedString> | ||||||
|  |  | ||||||
| #include <QtConcurrent/QtConcurrent> |  | ||||||
| #include <QBuffer> | #include <QBuffer> | ||||||
| #include <QGraphicsScene> | #include <QGraphicsScene> | ||||||
| #include <QGraphicsView> | #include <QGraphicsView> | ||||||
| #include <QPixmap> |  | ||||||
|  |  | ||||||
| #include <math.h> | #include <cmath> | ||||||
|  |  | ||||||
| #include "awactions.h" | #include "awactions.h" | ||||||
| #include "awdebug.h" | #include "awdebug.h" | ||||||
|  |  | ||||||
|  |  | ||||||
| AWDataAggregator::AWDataAggregator(QObject *parent) | AWDataAggregator::AWDataAggregator(QObject *_parent) | ||||||
|     : QObject(parent) |     : QObject(_parent) | ||||||
| { | { | ||||||
|     qCDebug(LOG_AW) << __PRETTY_FUNCTION__; |     qCDebug(LOG_AW) << __PRETTY_FUNCTION__; | ||||||
|     // required by signals |  | ||||||
|     qRegisterMetaType<QHash<QString, QString>>("QHash<QString, QString>"); |     m_boundaries["cpuTooltip"] = 100.0; | ||||||
|  |     m_boundaries["cpuclTooltip"] = 4000.0; | ||||||
|  |     m_boundaries["memTooltip"] = 100.0; | ||||||
|  |     m_boundaries["swapTooltip"] = 100.0; | ||||||
|  |     m_boundaries["downkbTooltip"] = 1.0; | ||||||
|  |     m_boundaries["upkbTooltip"] = 1.0; | ||||||
|  |     m_boundaries["batTooltip"] = 100.0; | ||||||
|  |  | ||||||
|     initScene(); |     initScene(); | ||||||
|     connect(this, SIGNAL(updateData(const QHash<QString, QString> &)), this, |  | ||||||
|             SLOT(dataUpdate(const QHash<QString, QString> &))); |  | ||||||
| } | } | ||||||
|  |  | ||||||
|  |  | ||||||
| @ -48,79 +50,59 @@ AWDataAggregator::~AWDataAggregator() | |||||||
| { | { | ||||||
|     qCDebug(LOG_AW) << __PRETTY_FUNCTION__; |     qCDebug(LOG_AW) << __PRETTY_FUNCTION__; | ||||||
|  |  | ||||||
|     delete toolTipScene; |     delete m_toolTipScene; | ||||||
| } | } | ||||||
|  |  | ||||||
|  |  | ||||||
| QList<float> AWDataAggregator::getData(const QString key) const | QString AWDataAggregator::htmlImage(const QPixmap &_source) | ||||||
| { |  | ||||||
|     qCDebug(LOG_AW) << "Key" << key; |  | ||||||
|  |  | ||||||
|     return data[QString("%1Tooltip").arg(key)]; |  | ||||||
| } |  | ||||||
|  |  | ||||||
|  |  | ||||||
| QString AWDataAggregator::htmlImage(const QPixmap &source) const |  | ||||||
| { | { | ||||||
|     QByteArray byteArray; |     QByteArray byteArray; | ||||||
|     QBuffer buffer(&byteArray); |     QBuffer buffer(&byteArray); | ||||||
|     source.save(&buffer, "PNG"); |     _source.save(&buffer, "PNG"); | ||||||
|  |  | ||||||
|     return byteArray.isEmpty() |     return byteArray.isEmpty() ? "" | ||||||
|                ? QString() |                                : QString("<img src=\"data:image/png;base64,%1\"/>").arg(QString(byteArray.toBase64())); | ||||||
|                : QString("<img src=\"data:image/png;base64,%1\"/>") |  | ||||||
|                      .arg(QString(byteArray.toBase64())); |  | ||||||
| } | } | ||||||
|  |  | ||||||
|  |  | ||||||
| void AWDataAggregator::setParameters(QVariantMap settings) | void AWDataAggregator::setParameters(const QVariantMap &_settings) | ||||||
| { | { | ||||||
|     qCDebug(LOG_AW) << "Settings" << settings; |     qCDebug(LOG_AW) << "Settings" << _settings; | ||||||
|  |  | ||||||
|     // cast from QVariantMap to QVariantHash without data lost |     // cast from QVariantMap to QVariantHash without data lost | ||||||
|     configuration = qvariant_cast<QVariantHash>(settings); |     m_configuration = qvariant_cast<QVariantHash>(_settings); | ||||||
|  |  | ||||||
|     m_enablePopup = configuration[QString("notify")].toBool(); |     m_enablePopup = m_configuration["notify"].toBool(); | ||||||
|  |  | ||||||
|     counts = 0; |     m_counts = 0; | ||||||
|     counts += configuration[QString("cpuTooltip")].toInt(); |     m_counts += m_configuration["cpuTooltip"].toInt(); | ||||||
|     counts += configuration[QString("cpuclTooltip")].toInt(); |     m_counts += m_configuration["cpuclTooltip"].toInt(); | ||||||
|     counts += configuration[QString("memTooltip")].toInt(); |     m_counts += m_configuration["memTooltip"].toInt(); | ||||||
|     counts += configuration[QString("swapTooltip")].toInt(); |     m_counts += m_configuration["swapTooltip"].toInt(); | ||||||
|     counts += configuration[QString("downTooltip")].toInt(); |     m_counts += m_configuration["downkbTooltip"].toInt(); | ||||||
|     counts += configuration[QString("batTooltip")].toInt(); |     m_counts += m_configuration["batTooltip"].toInt(); | ||||||
|     // resize tooltip image |     // resize tooltip image | ||||||
|     toolTipView->resize(100.0 * counts, 105.0); |     m_toolTipView->resize(100 * m_counts, 105); | ||||||
|  |  | ||||||
|     boundaries[QString("cpuTooltip")] = 100.0; |  | ||||||
|     boundaries[QString("cpuclTooltip")] = 4000.0; |  | ||||||
|     boundaries[QString("memTooltip")] = 100.0; |  | ||||||
|     boundaries[QString("swapTooltip")] = 100.0; |  | ||||||
|     boundaries[QString("downTooltip")] = 1.0; |  | ||||||
|     boundaries[QString("upTooltip")] = 1.0; |  | ||||||
|     boundaries[QString("batTooltip")] = 100.0; |  | ||||||
|  |  | ||||||
|     requiredKeys.clear(); |     requiredKeys.clear(); | ||||||
|     if (configuration[QString("cpuTooltip")].toBool()) |     if (m_configuration["cpuTooltip"].toBool()) | ||||||
|         requiredKeys.append(QString("cpuTooltip")); |         requiredKeys.append("cpuTooltip"); | ||||||
|     if (configuration[QString("cpuclTooltip")].toBool()) |     if (m_configuration["cpuclTooltip"].toBool()) | ||||||
|         requiredKeys.append(QString("cpuclTooltip")); |         requiredKeys.append("cpuclTooltip"); | ||||||
|     if (configuration[QString("memTooltip")].toBool()) |     if (m_configuration["memTooltip"].toBool()) | ||||||
|         requiredKeys.append(QString("memTooltip")); |         requiredKeys.append("memTooltip"); | ||||||
|     if (configuration[QString("swapTooltip")].toBool()) |     if (m_configuration["swapTooltip"].toBool()) | ||||||
|         requiredKeys.append(QString("swapTooltip")); |         requiredKeys.append("swapTooltip"); | ||||||
|     if (configuration[QString("downTooltip")].toBool()) |     if (m_configuration["downkbTooltip"].toBool()) | ||||||
|         requiredKeys.append(QString("downTooltip")); |         requiredKeys.append("downkbTooltip"); | ||||||
|     if (configuration[QString("upTooltip")].toBool()) |     if (m_configuration["upkbTooltip"].toBool()) | ||||||
|         requiredKeys.append(QString("upTooltip")); |         requiredKeys.append("upkbTooltip"); | ||||||
|     if (configuration[QString("batTooltip")].toBool()) |     if (m_configuration["batTooltip"].toBool()) | ||||||
|         requiredKeys.append(QString("batTooltip")); |         requiredKeys.append("batTooltip"); | ||||||
|  |  | ||||||
|     // background |     // background | ||||||
|     toolTipScene->setBackgroundBrush( |     m_toolTipScene->setBackgroundBrush(m_configuration["useTooltipBackground"].toBool() | ||||||
|         configuration[QString("useTooltipBackground")].toBool() |                                            ? QBrush(QColor(m_configuration["tooltipBackground"].toString())) | ||||||
|             ? QBrush(QColor( |  | ||||||
|                   configuration[QString("tooltipBackground")].toString())) |  | ||||||
|                                            : QBrush(Qt::NoBrush)); |                                            : QBrush(Qt::NoBrush)); | ||||||
| } | } | ||||||
|  |  | ||||||
| @ -128,199 +110,171 @@ void AWDataAggregator::setParameters(QVariantMap settings) | |||||||
| QPixmap AWDataAggregator::tooltipImage() | QPixmap AWDataAggregator::tooltipImage() | ||||||
| { | { | ||||||
|     // create image |     // create image | ||||||
|     toolTipScene->clear(); |     m_toolTipScene->clear(); | ||||||
|     QPen pen; |     QPen pen; | ||||||
|     bool down = false; |     bool down = false; | ||||||
|     foreach (QString key, requiredKeys) { |     for (auto &key : requiredKeys) { | ||||||
|         // create frame |         // create frame | ||||||
|         float normX = 100.0 / static_cast<float>(data[key].count()); |         float normX = 100.0f / static_cast<float>(m_values[key].count()); | ||||||
|         float normY = 100.0 / (1.5 * boundaries[key]); |         float normY = 100.0f / (1.5f * m_boundaries[key]); | ||||||
|         float shift = requiredKeys.indexOf(key) * 100.0; |         float shift = static_cast<float>(requiredKeys.indexOf(key)) * 100.0f; | ||||||
|         if (down) |         if (down) | ||||||
|             shift -= 100.0; |             shift -= 100.0; | ||||||
|         // apply pen color |         // apply pen color | ||||||
|         if (key != QString("batTooltip")) |         if (key != "batTooltip") | ||||||
|             pen.setColor( |             pen.setColor(QColor(m_configuration[QString("%1Color").arg(key)].toString())); | ||||||
|                 QColor(configuration[QString("%1Color").arg(key)].toString())); |  | ||||||
|         // paint data inside frame |         // paint data inside frame | ||||||
|         for (int j = 0; j < data[key].count() - 1; j++) { |         for (int j = 0; j < m_values[key].count() - 1; j++) { | ||||||
|             // some magic here |             // some magic here | ||||||
|             float x1 = j * normX + shift; |             float x1 = j * normX + shift; | ||||||
|             float y1 = -fabs(data[key].at(j)) * normY + 5.0; |             float y1 = -std::fabs(m_values[key].at(j)) * normY + 5.0f; | ||||||
|             float x2 = (j + 1) * normX + shift; |             float x2 = (j + 1) * normX + shift; | ||||||
|             float y2 = -fabs(data[key].at(j + 1)) * normY + 5.0; |             float y2 = -std::fabs(m_values[key].at(j + 1)) * normY + 5.0f; | ||||||
|             if (key == QString("batTooltip")) { |             if (key == "batTooltip") { | ||||||
|                 if (data[key].at(j + 1) > 0) |                 if (m_values[key].at(j + 1) > 0) | ||||||
|                     pen.setColor(QColor( |                     pen.setColor(QColor(m_configuration["batTooltipColor"].toString())); | ||||||
|                         configuration[QString("batTooltipColor")].toString())); |  | ||||||
|                 else |                 else | ||||||
|                     pen.setColor( |                     pen.setColor(QColor(m_configuration["batInTooltipColor"].toString())); | ||||||
|                         QColor(configuration[QString("batInTooltipColor")] |  | ||||||
|                                    .toString())); |  | ||||||
|             } |             } | ||||||
|             toolTipScene->addLine(x1, y1, x2, y2, pen); |             m_toolTipScene->addLine(x1, y1, x2, y2, pen); | ||||||
|         } |         } | ||||||
|         if (key == QString("downTooltip")) |         if (key == "downkbTooltip") | ||||||
|             down = true; |             down = true; | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     return toolTipView->grab(); |     return m_toolTipView->grab(); | ||||||
| } | } | ||||||
|  |  | ||||||
|  |  | ||||||
| void AWDataAggregator::dataUpdate(const QHash<QString, QString> &values) | void AWDataAggregator::dataUpdate(const QVariantHash &_values) | ||||||
| { | { | ||||||
|     // do not log these arguments |     // do not log these arguments | ||||||
|     setData(values); |     setData(_values); | ||||||
|     emit(toolTipPainted(htmlImage(tooltipImage()))); |     emit(toolTipPainted(htmlImage(tooltipImage()))); | ||||||
| } | } | ||||||
|  |  | ||||||
|  |  | ||||||
| void AWDataAggregator::checkValue(const QString source, const float value, | void AWDataAggregator::checkValue(const QString &_source, const float _value, const float _extremum) const | ||||||
|                                   const float extremum) const |  | ||||||
| { | { | ||||||
|     qCDebug(LOG_AW) << "Notification source" << source; |     qCDebug(LOG_AW) << "Notification source" << _source << "with value" << _value << "called with extremum" | ||||||
|     qCDebug(LOG_AW) << "Value" << value; |                     << _extremum; | ||||||
|     qCDebug(LOG_AW) << "Called with extremum" << extremum; |  | ||||||
|  |  | ||||||
|     if (value >= 0.0) { |     if (_value >= 0.0) { | ||||||
|         if ((m_enablePopup) && (value > extremum) |         if ((m_enablePopup) && (_value > _extremum) && (m_values[_source].last() < _extremum)) | ||||||
|             && (data[source].last() < extremum)) |             return AWActions::sendNotification("event", notificationText(_source, _value)); | ||||||
|             return AWActions::sendNotification(QString("event"), |  | ||||||
|                                                notificationText(source, value)); |  | ||||||
|     } else { |     } else { | ||||||
|         if ((m_enablePopup) && (value < extremum) |         if ((m_enablePopup) && (_value < _extremum) && (m_values[_source].last() > _extremum)) | ||||||
|             && (data[source].last() > extremum)) |             return AWActions::sendNotification("event", notificationText(_source, _value)); | ||||||
|             return AWActions::sendNotification(QString("event"), |  | ||||||
|                                                notificationText(source, value)); |  | ||||||
|     } |     } | ||||||
| } | } | ||||||
|  |  | ||||||
|  |  | ||||||
| void AWDataAggregator::checkValue(const QString source, const QString current, | void AWDataAggregator::checkValue(const QString &_source, const QString &_current, const QString &_received) const | ||||||
|                                   const QString received) const |  | ||||||
| { | { | ||||||
|     qCDebug(LOG_AW) << "Notification source" << source; |     qCDebug(LOG_AW) << "Notification source" << _source << "with current value" << _current << "and received one" | ||||||
|     qCDebug(LOG_AW) << "Current value" << current; |                     << _received; | ||||||
|     qCDebug(LOG_AW) << "Received value" << received; |  | ||||||
|  |  | ||||||
|     if ((m_enablePopup) && (current != received) && (!received.isEmpty())) |     if ((m_enablePopup) && (_current != _received) && (!_received.isEmpty())) | ||||||
|         return AWActions::sendNotification(QString("event"), |         return AWActions::sendNotification("event", notificationText(_source, _received)); | ||||||
|                                            notificationText(source, received)); |  | ||||||
| } | } | ||||||
|  |  | ||||||
|  |  | ||||||
| void AWDataAggregator::initScene() | void AWDataAggregator::initScene() | ||||||
| { | { | ||||||
|     toolTipScene = new QGraphicsScene(nullptr); |     m_toolTipScene = new QGraphicsScene(nullptr); | ||||||
|     toolTipView = new QGraphicsView(toolTipScene); |     m_toolTipView = new QGraphicsView(m_toolTipScene); | ||||||
|     toolTipView->setStyleSheet(QString("background: transparent")); |     m_toolTipView->setStyleSheet("background: transparent"); | ||||||
|     toolTipView->setContentsMargins(0, 0, 0, 0); |     m_toolTipView->setContentsMargins(0, 0, 0, 0); | ||||||
|     toolTipView->setFrameShape(QFrame::NoFrame); |     m_toolTipView->setFrameShape(QFrame::NoFrame); | ||||||
|     toolTipView->setHorizontalScrollBarPolicy(Qt::ScrollBarAlwaysOff); |     m_toolTipView->setHorizontalScrollBarPolicy(Qt::ScrollBarAlwaysOff); | ||||||
|     toolTipView->setVerticalScrollBarPolicy(Qt::ScrollBarAlwaysOff); |     m_toolTipView->setVerticalScrollBarPolicy(Qt::ScrollBarAlwaysOff); | ||||||
| } | } | ||||||
|  |  | ||||||
|  |  | ||||||
| QString AWDataAggregator::notificationText(const QString source, | QString AWDataAggregator::notificationText(const QString &_source, const float _value) | ||||||
|                                            const float value) const |  | ||||||
| { | { | ||||||
|     qCDebug(LOG_AW) << "Notification source" << source; |     qCDebug(LOG_AW) << "Notification source" << _source << "with value" << _value; | ||||||
|     qCDebug(LOG_AW) << "Value" << value; |  | ||||||
|  |  | ||||||
|     QString output; |     QString output; | ||||||
|     if (source == QString("batTooltip")) |     if (_source == "batTooltip") | ||||||
|         output = value > 0.0 ? i18n("AC online") : i18n("AC offline"); |         output = _value > 0.0 ? i18n("AC online") : i18n("AC offline"); | ||||||
|     else if (source == QString("cpuTooltip")) |     else if (_source == "cpuTooltip") | ||||||
|         output = i18n("High CPU load"); |         output = i18n("High CPU load"); | ||||||
|     else if (source == QString("memTooltip")) |     else if (_source == "memTooltip") | ||||||
|         output = i18n("High memory usage"); |         output = i18n("High memory usage"); | ||||||
|     else if (source == QString("swapTooltip")) |     else if (_source == "swapTooltip") | ||||||
|         output = i18n("Swap is used"); |         output = i18n("Swap is used"); | ||||||
|     else if (source == QString("gpu")) |     else if (_source == "gpu") | ||||||
|         output = i18n("High GPU load"); |         output = i18n("High GPU load"); | ||||||
|  |  | ||||||
|     return output; |     return output; | ||||||
| } | } | ||||||
|  |  | ||||||
|  |  | ||||||
| QString AWDataAggregator::notificationText(const QString source, | QString AWDataAggregator::notificationText(const QString &_source, const QString &_value) | ||||||
|                                            const QString value) const |  | ||||||
| { | { | ||||||
|     qCDebug(LOG_AW) << "Notification source" << source; |     qCDebug(LOG_AW) << "Notification source" << _source << "with value" << _value; | ||||||
|     qCDebug(LOG_AW) << "Value" << value; |  | ||||||
|  |  | ||||||
|     QString output; |     QString output; | ||||||
|     if (source == QString("netdev")) |     if (_source == "netdev") | ||||||
|         output = i18n("Network device has been changed to %1", value); |         output = i18n("Network device has been changed to %1", _value); | ||||||
|  |  | ||||||
|     return output; |     return output; | ||||||
| } | } | ||||||
|  |  | ||||||
|  |  | ||||||
| void AWDataAggregator::setData(const QHash<QString, QString> &values) | void AWDataAggregator::setData(const QVariantHash &_values) | ||||||
| { | { | ||||||
|     // do not log these arguments |     // do not log these arguments | ||||||
|     // battery update requires info is AC online or not |     // battery update requires info is AC online or not | ||||||
|     setData(values[QString("ac")] == configuration[QString("acOnline")], |     setData(_values["ac"].toString() == m_configuration["acOnline"], "batTooltip", _values["bat"].toFloat()); | ||||||
|             QString("batTooltip"), values[QString("bat")].toFloat()); |  | ||||||
|     // usual case |     // usual case | ||||||
|     setData(QString("cpuTooltip"), values[QString("cpu")].toFloat(), 90.0); |     setData("cpuTooltip", _values["cpu"].toFloat(), 90.0); | ||||||
|     setData(QString("cpuclTooltip"), values[QString("cpucl")].toFloat()); |     setData("cpuclTooltip", _values["cpucl"].toFloat()); | ||||||
|     setData(QString("memTooltip"), values[QString("mem")].toFloat(), 90.0); |     setData("memTooltip", _values["mem"].toFloat(), 80.0); | ||||||
|     setData(QString("swapTooltip"), values[QString("swap")].toFloat(), 0.0); |     setData("swapTooltip", _values["swap"].toFloat(), 0.0); | ||||||
|     setData(QString("downTooltip"), values[QString("downkb")].toFloat()); |     setData("downkbTooltip", _values["downkb"].toFloat()); | ||||||
|     setData(QString("upTooltip"), values[QString("upkb")].toFloat()); |     setData("upkbTooltip", _values["upkb"].toFloat()); | ||||||
|     // additional check for network device |     // additional check for network device | ||||||
|     [this](const QString value) { |     [this](const QString &value) { | ||||||
|         checkValue(QString("netdev"), currentNetworkDevice, value); |         checkValue("netdev", m_currentNetworkDevice, value); | ||||||
|         currentNetworkDevice = value; |         m_currentNetworkDevice = value; | ||||||
|     }(values[QString("netdev")]); |     }(_values["netdev"].toString()); | ||||||
|     // additional check for GPU load |     // additional check for GPU load | ||||||
|     [this](const float value) { |     [this](const float value) { checkValue("gpu", value, 90.0); }(_values["gpu"].toFloat()); | ||||||
|         checkValue(QString("gpu"), value, 90.0); |  | ||||||
|         currentGPULoad = value; |  | ||||||
|     }(values[QString("gpu")].toFloat()); |  | ||||||
| } | } | ||||||
|  |  | ||||||
|  |  | ||||||
| void AWDataAggregator::setData(const QString &source, float value, | void AWDataAggregator::setData(const QString &_source, float _value, const float _extremum) | ||||||
|                                const float extremum) |  | ||||||
| { | { | ||||||
|     qCDebug(LOG_AW) << "Source" << source; |     qCDebug(LOG_AW) << "Source" << _source << "to value" << _value << "with extremum" << _extremum; | ||||||
|     qCDebug(LOG_AW) << "Value" << value; |  | ||||||
|     qCDebug(LOG_AW) << "Called with extremum" << extremum; |  | ||||||
|  |  | ||||||
|     if (data[source].count() == 0) |     if (m_values[_source].count() == 0) | ||||||
|         data[source].append(0.0); |         m_values[_source].append(0.0); | ||||||
|     else if (data[source].count() |     else if (m_values[_source].count() > m_configuration["tooltipNumber"].toInt()) | ||||||
|              > configuration[QString("tooltipNumber")].toInt()) |         m_values[_source].removeFirst(); | ||||||
|         data[source].removeFirst(); |     if (std::isnan(_value)) | ||||||
|     if (isnan(value)) |         _value = 0.0; | ||||||
|         value = 0.0; |  | ||||||
|  |  | ||||||
|     // notifications |     // notifications | ||||||
|     checkValue(source, value, extremum); |     checkValue(_source, _value, _extremum); | ||||||
|  |  | ||||||
|     data[source].append(value); |     m_values[_source].append(_value); | ||||||
|     if (source == QString("downTooltip")) { |     if (_source == "downkbTooltip") { | ||||||
|         QList<float> netValues |         QList<float> netValues = m_values["downkbTooltip"] + m_values["upkbTooltip"]; | ||||||
|             = data[QString("downTooltip")] + data[QString("upTooltip")]; |         // to avoid inf value of normY | ||||||
|         boundaries[QString("downTooltip")] |         netValues << 1.0; | ||||||
|             = 1.2 * *std::max_element(netValues.cbegin(), netValues.cend()); |         m_boundaries["downkbTooltip"] = 1.2f * *std::max_element(netValues.cbegin(), netValues.cend()); | ||||||
|         boundaries[QString("upTooltip")] = boundaries[QString("downTooltip")]; |         m_boundaries["upkbTooltip"] = m_boundaries["downkbTooltip"]; | ||||||
|     } |     } | ||||||
| } | } | ||||||
|  |  | ||||||
|  |  | ||||||
| void AWDataAggregator::setData(const bool dontInvert, const QString &source, | void AWDataAggregator::setData(const bool _dontInvert, const QString &_source, float _value) | ||||||
|                                float value) |  | ||||||
| { | { | ||||||
|     qCDebug(LOG_AW) << "Do not invert value" << dontInvert; |     qCDebug(LOG_AW) << "Do not invert" << _dontInvert << "value" << _value << "for source" << _source; | ||||||
|     qCDebug(LOG_AW) << "Source" << source; |  | ||||||
|     qCDebug(LOG_AW) << "Value" << value; |  | ||||||
|  |  | ||||||
|     // invert values for different battery colours |     // invert values for different battery colours | ||||||
|     value = dontInvert ? value : -value; |     _value = _dontInvert ? _value : -_value; | ||||||
|     return setData(source, value, 0.0); |     return setData(_source, _value, 0.0); | ||||||
| } | } | ||||||
|  | |||||||
| @ -33,44 +33,38 @@ class AWDataAggregator : public QObject | |||||||
|     Q_OBJECT |     Q_OBJECT | ||||||
|  |  | ||||||
| public: | public: | ||||||
|     explicit AWDataAggregator(QObject *parent = nullptr); |     explicit AWDataAggregator(QObject *_parent = nullptr); | ||||||
|     virtual ~AWDataAggregator(); |     ~AWDataAggregator() override; | ||||||
|     QList<float> getData(const QString key) const; |     static QString htmlImage(const QPixmap &_source); | ||||||
|     QString htmlImage(const QPixmap &source) const; |     void setParameters(const QVariantMap &_settings); | ||||||
|     void setParameters(QVariantMap settings); |  | ||||||
|     QPixmap tooltipImage(); |     QPixmap tooltipImage(); | ||||||
|  |  | ||||||
| signals: |  | ||||||
|     void updateData(const QHash<QString, QString> &values); |  | ||||||
|     void toolTipPainted(const QString image) const; |  | ||||||
|  |  | ||||||
| public slots: | public slots: | ||||||
|     void dataUpdate(const QHash<QString, QString> &values); |     void dataUpdate(const QVariantHash &_values); | ||||||
|  |  | ||||||
|  | signals: | ||||||
|  |     void toolTipPainted(const QString &_image) const; | ||||||
|  |  | ||||||
| private: | private: | ||||||
|     // ui |     // ui | ||||||
|     QGraphicsScene *toolTipScene = nullptr; |     QGraphicsScene *m_toolTipScene = nullptr; | ||||||
|     QGraphicsView *toolTipView = nullptr; |     QGraphicsView *m_toolTipView = nullptr; | ||||||
|     void checkValue(const QString source, const float value, |     void checkValue(const QString &_source, float _value, float _extremum) const; | ||||||
|                     const float extremum) const; |     void checkValue(const QString &_source, const QString &_current, const QString &_received) const; | ||||||
|     void checkValue(const QString source, const QString current, |  | ||||||
|                     const QString received) const; |  | ||||||
|     void initScene(); |     void initScene(); | ||||||
|     QString notificationText(const QString source, const float value) const; |     static QString notificationText(const QString &_source, float _value); | ||||||
|     QString notificationText(const QString source, const QString value) const; |     static QString notificationText(const QString &_source, const QString &_value); | ||||||
|     // main method |     // main method | ||||||
|     void setData(const QHash<QString, QString> &values); |     void setData(const QVariantHash &_values); | ||||||
|     void setData(const QString &source, float value, |     void setData(const QString &_source, float _value, float _extremum = -1.0f); | ||||||
|                  const float extremum = -1.0); |  | ||||||
|     // different signature for battery device |     // different signature for battery device | ||||||
|     void setData(const bool dontInvert, const QString &source, float value); |     void setData(bool _dontInvert, const QString &_source, float _value); | ||||||
|     // variables |     // variables | ||||||
|     int counts = 0; |     int m_counts = 0; | ||||||
|     QVariantHash configuration; |     QVariantHash m_configuration; | ||||||
|     float currentGPULoad = 0.0; |     QString m_currentNetworkDevice = "lo"; | ||||||
|     QString currentNetworkDevice = QString("lo"); |     QHash<QString, float> m_boundaries; | ||||||
|     QHash<QString, float> boundaries; |     QHash<QString, QList<float>> m_values; | ||||||
|     QHash<QString, QList<float>> data; |  | ||||||
|     bool m_enablePopup = false; |     bool m_enablePopup = false; | ||||||
|     QStringList requiredKeys; |     QStringList requiredKeys; | ||||||
| }; | }; | ||||||
|  | |||||||
| @ -17,20 +17,30 @@ | |||||||
|  |  | ||||||
| #include "awdataengineaggregator.h" | #include "awdataengineaggregator.h" | ||||||
|  |  | ||||||
| #include <Plasma/DataEngineConsumer> | #include <Plasma/DataContainer> | ||||||
|  |  | ||||||
| #include "awdebug.h" | #include "awdebug.h" | ||||||
| #include "awkeys.h" |  | ||||||
|  |  | ||||||
|  |  | ||||||
| AWDataEngineAggregator::AWDataEngineAggregator(QObject *parent, | AWDataEngineAggregator::AWDataEngineAggregator(QObject *_parent) | ||||||
|                                                const int interval) |     : QObject(_parent) | ||||||
|     : QObject(parent) |  | ||||||
| { | { | ||||||
|     qCDebug(LOG_AW) << __PRETTY_FUNCTION__; |     qCDebug(LOG_AW) << __PRETTY_FUNCTION__; | ||||||
|  |  | ||||||
|     setInterval(interval); |     m_consumer = new Plasma::DataEngineConsumer(); | ||||||
|     initDataEngines(); |     m_dataEngines["systemmonitor"] = m_consumer->dataEngine("systemmonitor"); | ||||||
|  |     m_dataEngines["extsysmon"] = m_consumer->dataEngine("extsysmon"); | ||||||
|  |     m_dataEngines["time"] = m_consumer->dataEngine("time"); | ||||||
|  |  | ||||||
|  |     // additional method required by systemmonitor structure | ||||||
|  |     m_newSourceConnection | ||||||
|  |         = connect(m_dataEngines["systemmonitor"], &Plasma::DataEngine::sourceAdded, [this](const QString &source) { | ||||||
|  |               emit(deviceAdded(source)); | ||||||
|  |               m_dataEngines["systemmonitor"]->connectSource(source, parent(), 1000); | ||||||
|  |           }); | ||||||
|  |  | ||||||
|  |     // required to define Qt::QueuedConnection for signal-slot connection | ||||||
|  |     qRegisterMetaType<Plasma::DataEngine::Data>("Plasma::DataEngine::Data"); | ||||||
| } | } | ||||||
|  |  | ||||||
|  |  | ||||||
| @ -38,64 +48,68 @@ AWDataEngineAggregator::~AWDataEngineAggregator() | |||||||
| { | { | ||||||
|     qCDebug(LOG_AW) << __PRETTY_FUNCTION__; |     qCDebug(LOG_AW) << __PRETTY_FUNCTION__; | ||||||
|  |  | ||||||
|     // disconnect sources first |  | ||||||
|     disconnectSources(); |     disconnectSources(); | ||||||
|     m_dataEngines.clear(); |  | ||||||
| } | } | ||||||
|  |  | ||||||
|  |  | ||||||
| void AWDataEngineAggregator::disconnectSources() | void AWDataEngineAggregator::disconnectSources() | ||||||
| { | { | ||||||
|     foreach (QString dataengine, m_dataEngines.keys()) |     for (auto dataEngine : m_dataEngines.values()) | ||||||
|         foreach (QString source, m_dataEngines[dataengine]->sources()) |         for (auto &source : dataEngine->sources()) | ||||||
|             m_dataEngines[dataengine]->disconnectSource(source, parent()); |             dataEngine->disconnectSource(source, parent()); | ||||||
|  |     disconnect(m_newSourceConnection); | ||||||
| } | } | ||||||
|  |  | ||||||
|  |  | ||||||
| void AWDataEngineAggregator::setInterval(const int _interval) | void AWDataEngineAggregator::reconnectSources(const int _interval) | ||||||
| { | { | ||||||
|     qCDebug(LOG_AW) << "Interval" << _interval; |     qCDebug(LOG_AW) << "Reconnect sources with interval" << _interval; | ||||||
|  |  | ||||||
|     m_interval = _interval; |     disconnectSources(); | ||||||
| } |  | ||||||
|  |  | ||||||
|  |     m_dataEngines["systemmonitor"]->connectAllSources(parent(), (uint)_interval); | ||||||
|  |     m_dataEngines["extsysmon"]->connectAllSources(parent(), (uint)_interval); | ||||||
|  |     m_dataEngines["time"]->connectSource("Local", parent(), 1000); | ||||||
|  |  | ||||||
| void AWDataEngineAggregator::dropSource(const QString source) |     m_newSourceConnection = connect( | ||||||
| { |         m_dataEngines["systemmonitor"], &Plasma::DataEngine::sourceAdded, [this, _interval](const QString &source) { | ||||||
|     qCDebug(LOG_AW) << "Source" << source; |             emit(deviceAdded(source)); | ||||||
|  |             m_dataEngines["systemmonitor"]->connectSource(source, parent(), (uint)_interval); | ||||||
|     // FIXME there is no possibility to check to which dataengine source |  | ||||||
|     // connected we will try to disconnect it from systemmonitor and extsysmon |  | ||||||
|     m_dataEngines[QString("systemmonitor")]->disconnectSource(source, parent()); |  | ||||||
|     m_dataEngines[QString("extsysmon")]->disconnectSource(source, parent()); |  | ||||||
| } |  | ||||||
|  |  | ||||||
|  |  | ||||||
| void AWDataEngineAggregator::reconnectSources() |  | ||||||
| { |  | ||||||
|     m_dataEngines[QString("systemmonitor")]->connectAllSources(parent(), |  | ||||||
|                                                                m_interval); |  | ||||||
|     m_dataEngines[QString("extsysmon")]->connectAllSources(parent(), |  | ||||||
|                                                            m_interval); |  | ||||||
|     m_dataEngines[QString("time")]->connectSource(QString("Local"), parent(), |  | ||||||
|                                                   1000); |  | ||||||
| } |  | ||||||
|  |  | ||||||
|  |  | ||||||
| void AWDataEngineAggregator::initDataEngines() |  | ||||||
| { |  | ||||||
|     Plasma::DataEngineConsumer *deConsumer = new Plasma::DataEngineConsumer(); |  | ||||||
|     m_dataEngines[QString("systemmonitor")] |  | ||||||
|         = deConsumer->dataEngine(QString("systemmonitor")); |  | ||||||
|     m_dataEngines[QString("extsysmon")] |  | ||||||
|         = deConsumer->dataEngine(QString("extsysmon")); |  | ||||||
|     m_dataEngines[QString("time")] = deConsumer->dataEngine(QString("time")); |  | ||||||
|  |  | ||||||
|     // additional method required by systemmonitor structure |  | ||||||
|     connect(m_dataEngines[QString("systemmonitor")], |  | ||||||
|             &Plasma::DataEngine::sourceAdded, [this](const QString source) { |  | ||||||
|                 static_cast<AWKeys *>(parent())->addDevice(source); |  | ||||||
|                 m_dataEngines[QString("systemmonitor")]->connectSource( |  | ||||||
|                     source, parent(), m_interval); |  | ||||||
|         }); |         }); | ||||||
|  |  | ||||||
|  | #ifdef BUILD_FUTURE | ||||||
|  |     createQueuedConnection(); | ||||||
|  | #endif /* BUILD_FUTURE */ | ||||||
|  | } | ||||||
|  |  | ||||||
|  |  | ||||||
|  | void AWDataEngineAggregator::dropSource(const QString &_source) | ||||||
|  | { | ||||||
|  |     qCDebug(LOG_AW) << "Source" << _source; | ||||||
|  |  | ||||||
|  |     // HACK there is no possibility to check to which dataengine source | ||||||
|  |     // connected we will try to disconnect it from all engines | ||||||
|  |     for (auto dataEngine : m_dataEngines.values()) | ||||||
|  |         dataEngine->disconnectSource(_source, parent()); | ||||||
|  | } | ||||||
|  |  | ||||||
|  |  | ||||||
|  | void AWDataEngineAggregator::createQueuedConnection() | ||||||
|  | { | ||||||
|  |     // HACK additional method which forces QueuedConnection instead of Auto one | ||||||
|  |     // for more details refer to plasma-framework source code | ||||||
|  |     for (auto &dataEngine : m_dataEngines.keys()) { | ||||||
|  |         // different source set for different engines | ||||||
|  |         QStringList sources = dataEngine == "time" ? QStringList() << "Local" : m_dataEngines[dataEngine]->sources(); | ||||||
|  |         // reconnect sources | ||||||
|  |         for (auto &source : sources) { | ||||||
|  |             Plasma::DataContainer *container = m_dataEngines[dataEngine]->containerForSource(source); | ||||||
|  |             // disconnect old connections first | ||||||
|  |             disconnect(container, SIGNAL(dataUpdated(QString, Plasma::DataEngine::Data)), parent(), | ||||||
|  |                        SLOT(dataUpdated(QString, Plasma::DataEngine::Data))); | ||||||
|  |             // and now reconnect with Qt::QueuedConnection type | ||||||
|  |             connect(container, SIGNAL(dataUpdated(QString, Plasma::DataEngine::Data)), parent(), | ||||||
|  |                     SLOT(dataUpdated(QString, Plasma::DataEngine::Data)), Qt::QueuedConnection); | ||||||
|  |         } | ||||||
|  |     } | ||||||
| } | } | ||||||
|  | |||||||
| @ -20,6 +20,7 @@ | |||||||
| #define AWDATAENGINEAGGREGATOR_H | #define AWDATAENGINEAGGREGATOR_H | ||||||
|  |  | ||||||
| #include <Plasma/DataEngine> | #include <Plasma/DataEngine> | ||||||
|  | #include <Plasma/DataEngineConsumer> | ||||||
|  |  | ||||||
| #include <QObject> | #include <QObject> | ||||||
|  |  | ||||||
| @ -27,24 +28,24 @@ | |||||||
| class AWDataEngineAggregator : public QObject | class AWDataEngineAggregator : public QObject | ||||||
| { | { | ||||||
|     Q_OBJECT |     Q_OBJECT | ||||||
|     Q_PROPERTY(int interval MEMBER m_interval WRITE setInterval); |  | ||||||
|  |  | ||||||
| public: | public: | ||||||
|     explicit AWDataEngineAggregator(QObject *parent = nullptr, |     explicit AWDataEngineAggregator(QObject *_parent = nullptr); | ||||||
|                                     const int interval = 1000); |     ~AWDataEngineAggregator() override; | ||||||
|     virtual ~AWDataEngineAggregator(); |  | ||||||
|     void disconnectSources(); |     void disconnectSources(); | ||||||
|     // properties |     void reconnectSources(int _interval); | ||||||
|     void setInterval(const int _interval); |  | ||||||
|  | signals: | ||||||
|  |     void deviceAdded(const QString &_source); | ||||||
|  |  | ||||||
| public slots: | public slots: | ||||||
|     void dropSource(const QString source); |     void dropSource(const QString &_source); | ||||||
|     void reconnectSources(); |  | ||||||
|  |  | ||||||
| private: | private: | ||||||
|     void initDataEngines(); |     void createQueuedConnection(); | ||||||
|  |     Plasma::DataEngineConsumer *m_consumer = nullptr; | ||||||
|     QHash<QString, Plasma::DataEngine *> m_dataEngines; |     QHash<QString, Plasma::DataEngine *> m_dataEngines; | ||||||
|     int m_interval; |     QMetaObject::Connection m_newSourceConnection; | ||||||
| }; | }; | ||||||
|  |  | ||||||
|  |  | ||||||
|  | |||||||
							
								
								
									
										445
									
								
								sources/awesome-widget/plugin/awdataenginemapper.cpp
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										445
									
								
								sources/awesome-widget/plugin/awdataenginemapper.cpp
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,445 @@ | |||||||
|  | /*************************************************************************** | ||||||
|  |  *   This file is part of awesome-widgets                                  * | ||||||
|  |  *                                                                         * | ||||||
|  |  *   awesome-widgets is free software: you can redistribute it and/or      * | ||||||
|  |  *   modify it under the terms of the GNU General Public License as        * | ||||||
|  |  *   published by the Free Software Foundation, either version 3 of the    * | ||||||
|  |  *   License, or (at your option) any later version.                       * | ||||||
|  |  *                                                                         * | ||||||
|  |  *   awesome-widgets is distributed in the hope that it will be useful,    * | ||||||
|  |  *   but WITHOUT ANY WARRANTY; without even the implied warranty of        * | ||||||
|  |  *   MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the         * | ||||||
|  |  *   GNU General Public License for more details.                          * | ||||||
|  |  *                                                                         * | ||||||
|  |  *   You should have received a copy of the GNU General Public License     * | ||||||
|  |  *   along with awesome-widgets. If not, see http://www.gnu.org/licenses/  * | ||||||
|  |  ***************************************************************************/ | ||||||
|  |  | ||||||
|  | #include "awdataenginemapper.h" | ||||||
|  |  | ||||||
|  | #include <QRegExp> | ||||||
|  |  | ||||||
|  | #include "awdebug.h" | ||||||
|  | #include "awformatterhelper.h" | ||||||
|  |  | ||||||
|  |  | ||||||
|  | AWDataEngineMapper::AWDataEngineMapper(QObject *_parent, AWFormatterHelper *_custom) | ||||||
|  |     : QObject(_parent) | ||||||
|  |     , m_customFormatters(_custom) | ||||||
|  | { | ||||||
|  |     qCDebug(LOG_AW) << __PRETTY_FUNCTION__; | ||||||
|  |  | ||||||
|  |     // default formatters | ||||||
|  |     // memory | ||||||
|  |     m_formatter["mem"] = AWKeysAggregator::FormatterType::Float; | ||||||
|  |     m_formatter["memtotmb"] = AWKeysAggregator::FormatterType::MemMBFormat; | ||||||
|  |     m_formatter["memtotgb"] = AWKeysAggregator::FormatterType::MemGBFormat; | ||||||
|  |     // network | ||||||
|  |     m_formatter["down"] = AWKeysAggregator::FormatterType::NetSmartFormat; | ||||||
|  |     m_formatter["downkb"] = AWKeysAggregator::FormatterType::Integer; | ||||||
|  |     m_formatter["downtot"] = AWKeysAggregator::FormatterType::MemMBFormat; | ||||||
|  |     m_formatter["downtotkb"] = AWKeysAggregator::FormatterType::Integer; | ||||||
|  |     m_formatter["downunits"] = AWKeysAggregator::FormatterType::NetSmartUnits; | ||||||
|  |     m_formatter["up"] = AWKeysAggregator::FormatterType::NetSmartFormat; | ||||||
|  |     m_formatter["upkb"] = AWKeysAggregator::FormatterType::Integer; | ||||||
|  |     m_formatter["uptot"] = AWKeysAggregator::FormatterType::MemMBFormat; | ||||||
|  |     m_formatter["uptotkb"] = AWKeysAggregator::FormatterType::Integer; | ||||||
|  |     m_formatter["upunits"] = AWKeysAggregator::FormatterType::NetSmartUnits; | ||||||
|  |     // swap | ||||||
|  |     m_formatter["swap"] = AWKeysAggregator::FormatterType::Float; | ||||||
|  |     m_formatter["swaptotmb"] = AWKeysAggregator::FormatterType::MemMBFormat; | ||||||
|  |     m_formatter["swaptotgb"] = AWKeysAggregator::FormatterType::MemGBFormat; | ||||||
|  | } | ||||||
|  |  | ||||||
|  |  | ||||||
|  | AWDataEngineMapper::~AWDataEngineMapper() | ||||||
|  | { | ||||||
|  |     qCDebug(LOG_AW) << __PRETTY_FUNCTION__; | ||||||
|  | } | ||||||
|  |  | ||||||
|  |  | ||||||
|  | AWKeysAggregator::FormatterType AWDataEngineMapper::formatter(const QString &_key) const | ||||||
|  | { | ||||||
|  |     qCDebug(LOG_AW) << "Get formatter for key" << _key; | ||||||
|  |  | ||||||
|  |     return m_formatter.value(_key, AWKeysAggregator::FormatterType::NoFormat); | ||||||
|  | } | ||||||
|  |  | ||||||
|  |  | ||||||
|  | QStringList AWDataEngineMapper::keysFromSource(const QString &_source) const | ||||||
|  | { | ||||||
|  |     qCDebug(LOG_AW) << "Search for source" << _source; | ||||||
|  |  | ||||||
|  |     return m_map.values(_source); | ||||||
|  | } | ||||||
|  |  | ||||||
|  |  | ||||||
|  | // HACK units required to define should the value be calculated as temperature | ||||||
|  | // or fan data | ||||||
|  | QStringList AWDataEngineMapper::registerSource(const QString &_source, const QString &_units, const QStringList &_keys) | ||||||
|  | { | ||||||
|  |     qCDebug(LOG_AW) << "Source" << _source << "with units" << _units; | ||||||
|  |  | ||||||
|  |     // regular expressions | ||||||
|  |     QRegExp cpuRegExp = QRegExp("cpu/cpu.*/TotalLoad"); | ||||||
|  |     QRegExp cpuclRegExp = QRegExp("cpu/cpu.*/clock"); | ||||||
|  |     QRegExp hddrRegExp = QRegExp("disk/.*/Rate/rblk"); | ||||||
|  |     QRegExp hddwRegExp = QRegExp("disk/.*/Rate/wblk"); | ||||||
|  |     QRegExp mountFillRegExp = QRegExp("partitions/.*/filllevel"); | ||||||
|  |     QRegExp mountFreeRegExp = QRegExp("partitions/.*/freespace"); | ||||||
|  |     QRegExp mountUsedRegExp = QRegExp("partitions/.*/usedspace"); | ||||||
|  |     QRegExp netRegExp = QRegExp("network/interfaces/.*/(receiver|transmitter)/data$"); | ||||||
|  |     QRegExp netTotalRegExp = QRegExp("network/interfaces/.*/(receiver|transmitter)/dataTotal$"); | ||||||
|  |  | ||||||
|  |     if (_source == "battery/ac") { | ||||||
|  |         // AC | ||||||
|  |         m_map.insert(_source, "ac"); | ||||||
|  |         m_formatter["ac"] = AWKeysAggregator::FormatterType::ACFormat; | ||||||
|  |     } else if (_source.startsWith("battery/")) { | ||||||
|  |         // battery stats | ||||||
|  |         QString key = _source; | ||||||
|  |         key.remove("battery/"); | ||||||
|  |         m_map.insert(_source, key); | ||||||
|  |         m_formatter[key] = _source.contains("rate") ? AWKeysAggregator::FormatterType::Float | ||||||
|  |                                                     : AWKeysAggregator::FormatterType::IntegerThree; | ||||||
|  |     } else if (_source == "cpu/system/TotalLoad") { | ||||||
|  |         // cpu | ||||||
|  |         m_map.insert(_source, "cpu"); | ||||||
|  |         m_formatter["cpu"] = AWKeysAggregator::FormatterType::Float; | ||||||
|  |     } else if (_source.contains(cpuRegExp)) { | ||||||
|  |         // cpus | ||||||
|  |         QString key = _source; | ||||||
|  |         key.remove("cpu/").remove("/TotalLoad"); | ||||||
|  |         m_map.insert(_source, key); | ||||||
|  |         m_formatter[key] = AWKeysAggregator::FormatterType::Float; | ||||||
|  |     } else if (_source == "cpu/system/AverageClock") { | ||||||
|  |         // cpucl | ||||||
|  |         m_map.insert(_source, "cpucl"); | ||||||
|  |         m_formatter["cpucl"] = AWKeysAggregator::FormatterType::Integer; | ||||||
|  |     } else if (_source.contains(cpuclRegExp)) { | ||||||
|  |         // cpucls | ||||||
|  |         QString key = _source; | ||||||
|  |         key.remove("cpu/cpu").remove("/clock"); | ||||||
|  |         key = QString("cpucl%1").arg(key); | ||||||
|  |         m_map.insert(_source, key); | ||||||
|  |         m_formatter[key] = AWKeysAggregator::FormatterType::Integer; | ||||||
|  |     } else if (_source.startsWith("custom")) { | ||||||
|  |         // custom | ||||||
|  |         QString key = _source; | ||||||
|  |         key.remove("custom/"); | ||||||
|  |         m_map.insert(_source, key); | ||||||
|  |         m_formatter[key] = AWKeysAggregator::FormatterType::NoFormat; | ||||||
|  |     } else if (_source == "desktop/current/name") { | ||||||
|  |         // current desktop name | ||||||
|  |         m_map.insert(_source, "desktop"); | ||||||
|  |         m_formatter["desktop"] = AWKeysAggregator::FormatterType::NoFormat; | ||||||
|  |     } else if (_source == "desktop/current/number") { | ||||||
|  |         // current desktop number | ||||||
|  |         m_map.insert(_source, "ndesktop"); | ||||||
|  |         m_formatter["ndesktop"] = AWKeysAggregator::FormatterType::NoFormat; | ||||||
|  |     } else if (_source == "desktop/total/number") { | ||||||
|  |         // desktop count | ||||||
|  |         m_map.insert(_source, "tdesktops"); | ||||||
|  |         m_formatter["tdesktops"] = AWKeysAggregator::FormatterType::NoFormat; | ||||||
|  |     } else if (_source.contains(hddrRegExp)) { | ||||||
|  |         // read speed | ||||||
|  |         QString device = _source; | ||||||
|  |         device.remove("/Rate/rblk"); | ||||||
|  |         int index = m_devices["disk"].indexOf(device); | ||||||
|  |         if (index > -1) { | ||||||
|  |             QString key = QString("hddr%1").arg(index); | ||||||
|  |             m_map.insert(_source, key); | ||||||
|  |             m_formatter[key] = AWKeysAggregator::FormatterType::Integer; | ||||||
|  |         } | ||||||
|  |     } else if (_source.contains(hddwRegExp)) { | ||||||
|  |         // write speed | ||||||
|  |         QString device = _source; | ||||||
|  |         device.remove("/Rate/wblk"); | ||||||
|  |         int index = m_devices["disk"].indexOf(device); | ||||||
|  |         if (index > -1) { | ||||||
|  |             QString key = QString("hddw%1").arg(index); | ||||||
|  |             m_map.insert(_source, key); | ||||||
|  |             m_formatter[key] = AWKeysAggregator::FormatterType::Integer; | ||||||
|  |         } | ||||||
|  |     } else if (_source == "gpu/load") { | ||||||
|  |         // gpu load | ||||||
|  |         m_map.insert(_source, "gpu"); | ||||||
|  |         m_formatter["gpu"] = AWKeysAggregator::FormatterType::Float; | ||||||
|  |     } else if (_source == "gpu/temperature") { | ||||||
|  |         // gpu temperature | ||||||
|  |         m_map.insert(_source, "gputemp"); | ||||||
|  |         m_formatter["gputemp"] = AWKeysAggregator::FormatterType::Temperature; | ||||||
|  |     } else if (_source.contains(mountFillRegExp)) { | ||||||
|  |         // fill level | ||||||
|  |         QString device = _source; | ||||||
|  |         device.remove("partitions").remove("/filllevel"); | ||||||
|  |         int index = m_devices["mount"].indexOf(device); | ||||||
|  |         if (index > -1) { | ||||||
|  |             QString key = QString("hdd%1").arg(index); | ||||||
|  |             m_map.insert(_source, key); | ||||||
|  |             m_formatter[key] = AWKeysAggregator::FormatterType::Float; | ||||||
|  |             // additional keys | ||||||
|  |             m_formatter[QString("hddtotmb%1").arg(index)] = AWKeysAggregator::FormatterType::MemMBFormat; | ||||||
|  |             m_formatter[QString("hddtotgb%1").arg(index)] = AWKeysAggregator::FormatterType::MemGBFormat; | ||||||
|  |         } | ||||||
|  |     } else if (_source.contains(mountFreeRegExp)) { | ||||||
|  |         // free space | ||||||
|  |         QString device = _source; | ||||||
|  |         device.remove("partitions").remove("/freespace"); | ||||||
|  |         int index = m_devices["mount"].indexOf(device); | ||||||
|  |         if (index > -1) { | ||||||
|  |             // mb | ||||||
|  |             QString key = QString("hddfreemb%1").arg(index); | ||||||
|  |             m_map.insert(_source, key); | ||||||
|  |             m_formatter[key] = AWKeysAggregator::FormatterType::MemMBFormat; | ||||||
|  |             // gb | ||||||
|  |             key = QString("hddfreegb%1").arg(index); | ||||||
|  |             m_map.insert(_source, key); | ||||||
|  |             m_formatter[key] = AWKeysAggregator::FormatterType::MemGBFormat; | ||||||
|  |         } | ||||||
|  |     } else if (_source.contains(mountUsedRegExp)) { | ||||||
|  |         // used | ||||||
|  |         QString device = _source; | ||||||
|  |         device.remove("partitions").remove("/usedspace"); | ||||||
|  |         int index = m_devices["mount"].indexOf(device); | ||||||
|  |         if (index > -1) { | ||||||
|  |             // mb | ||||||
|  |             QString key = QString("hddmb%1").arg(index); | ||||||
|  |             m_map.insert(_source, key); | ||||||
|  |             m_formatter[key] = AWKeysAggregator::FormatterType::MemMBFormat; | ||||||
|  |             // gb | ||||||
|  |             key = QString("hddgb%1").arg(index); | ||||||
|  |             m_map.insert(_source, key); | ||||||
|  |             m_formatter[key] = AWKeysAggregator::FormatterType::MemGBFormat; | ||||||
|  |         } | ||||||
|  |     } else if (_source.startsWith("hdd/temperature")) { | ||||||
|  |         // hdd temperature | ||||||
|  |         QString device = _source; | ||||||
|  |         device.remove("hdd/temperature"); | ||||||
|  |         int index = m_devices["hdd"].indexOf(device); | ||||||
|  |         if (index > -1) { | ||||||
|  |             QString key = QString("hddtemp%1").arg(index); | ||||||
|  |             m_map.insert(_source, key); | ||||||
|  |             m_formatter[key] = AWKeysAggregator::FormatterType::Temperature; | ||||||
|  |         } | ||||||
|  |     } else if (_source.startsWith("cpu/system/loadavg")) { | ||||||
|  |         // load average | ||||||
|  |         QString time = _source; | ||||||
|  |         time.remove("cpu/system/loadavg"); | ||||||
|  |         QString key = QString("la%1").arg(time); | ||||||
|  |         m_map.insert(_source, key); | ||||||
|  |         m_formatter[key] = AWKeysAggregator::FormatterType::FloatTwoSymbols; | ||||||
|  |     } else if (_source == "mem/physical/application") { | ||||||
|  |         // app memory | ||||||
|  |         // mb | ||||||
|  |         m_map.insert(_source, "memmb"); | ||||||
|  |         m_formatter["memmb"] = AWKeysAggregator::FormatterType::MemMBFormat; | ||||||
|  |         // gb | ||||||
|  |         m_map.insert(_source, "memgb"); | ||||||
|  |         m_formatter["memgb"] = AWKeysAggregator::FormatterType::MemGBFormat; | ||||||
|  |     } else if (_source == "mem/physical/free") { | ||||||
|  |         // free memory | ||||||
|  |         // mb | ||||||
|  |         m_map.insert(_source, "memfreemb"); | ||||||
|  |         m_formatter["memfreemb"] = AWKeysAggregator::FormatterType::MemMBFormat; | ||||||
|  |         // gb | ||||||
|  |         m_map.insert(_source, "memfreegb"); | ||||||
|  |         m_formatter["memfreegb"] = AWKeysAggregator::FormatterType::MemGBFormat; | ||||||
|  |     } else if (_source == "mem/physical/used") { | ||||||
|  |         // used memory | ||||||
|  |         // mb | ||||||
|  |         m_map.insert(_source, "memusedmb"); | ||||||
|  |         m_formatter["memusedmb"] = AWKeysAggregator::FormatterType::MemMBFormat; | ||||||
|  |         // gb | ||||||
|  |         m_map.insert(_source, "memusedgb"); | ||||||
|  |         m_formatter["memusedgb"] = AWKeysAggregator::FormatterType::MemGBFormat; | ||||||
|  |     } else if (_source == "network/current/name") { | ||||||
|  |         // network device | ||||||
|  |         m_map.insert(_source, "netdev"); | ||||||
|  |         m_formatter["netdev"] = AWKeysAggregator::FormatterType::NoFormat; | ||||||
|  |     } else if (_source == "network/current/ssid") { | ||||||
|  |         // current ssid | ||||||
|  |         m_map.insert(_source, "ssid"); | ||||||
|  |         m_formatter["ssid"] = AWKeysAggregator::FormatterType::NoFormat; | ||||||
|  |     } else if (_source.startsWith("network/response")) { | ||||||
|  |         // network response | ||||||
|  |         QString key = _source; | ||||||
|  |         key.remove("network/"); | ||||||
|  |         m_map.insert(_source, key); | ||||||
|  |         m_formatter[key] = AWKeysAggregator::FormatterType::NoFormat; | ||||||
|  |     } else if (_source.contains(netRegExp)) { | ||||||
|  |         // network speed | ||||||
|  |         QString type = _source.contains("receiver") ? "down" : "up"; | ||||||
|  |         int index = m_devices["net"].indexOf(_source.split('/')[2]); | ||||||
|  |         if (index > -1) { | ||||||
|  |             // kb | ||||||
|  |             QString key = QString("%1kb%2").arg(type).arg(index); | ||||||
|  |             m_map.insert(_source, key); | ||||||
|  |             m_formatter[key] = AWKeysAggregator::FormatterType::Integer; | ||||||
|  |             // smart | ||||||
|  |             key = QString("%1%2").arg(type).arg(index); | ||||||
|  |             m_map.insert(_source, key); | ||||||
|  |             m_formatter[key] = AWKeysAggregator::FormatterType::NetSmartFormat; | ||||||
|  |             // units | ||||||
|  |             key = QString("%1units%2").arg(type).arg(index); | ||||||
|  |             m_map.insert(_source, key); | ||||||
|  |             m_formatter[key] = AWKeysAggregator::FormatterType::NetSmartUnits; | ||||||
|  |         } | ||||||
|  |     } else if (_source.contains(netTotalRegExp)) { | ||||||
|  |         // network data total | ||||||
|  |         QString type = _source.contains("receiver") ? "down" : "up"; | ||||||
|  |         int index = m_devices["net"].indexOf(_source.split('/')[2]); | ||||||
|  |         if (index > -1) { | ||||||
|  |             // kb | ||||||
|  |             QString key = QString("%1totkb%2").arg(type).arg(index); | ||||||
|  |             m_map.insert(_source, key); | ||||||
|  |             m_formatter[key] = AWKeysAggregator::FormatterType::Integer; | ||||||
|  |             // mb | ||||||
|  |             key = QString("%1tot%2").arg(type).arg(index); | ||||||
|  |             m_map.insert(_source, key); | ||||||
|  |             m_formatter[key] = AWKeysAggregator::FormatterType::MemMBFormat; | ||||||
|  |         } | ||||||
|  |     } else if (_source.startsWith("upgrade")) { | ||||||
|  |         // package manager | ||||||
|  |         QString key = _source; | ||||||
|  |         key.remove("upgrade/"); | ||||||
|  |         m_map.insert(_source, key); | ||||||
|  |         m_formatter[key] = AWKeysAggregator::FormatterType::IntegerThree; | ||||||
|  |     } else if (_source.startsWith("player")) { | ||||||
|  |         // player | ||||||
|  |         QString key = _source; | ||||||
|  |         key.remove("player/"); | ||||||
|  |         m_map.insert(_source, key); | ||||||
|  |         m_formatter[key] = AWKeysAggregator::FormatterType::NoFormat; | ||||||
|  |     } else if (_source == "ps/running/count") { | ||||||
|  |         // running processes count | ||||||
|  |         m_map.insert(_source, "pscount"); | ||||||
|  |         m_formatter["pscount"] = AWKeysAggregator::FormatterType::NoFormat; | ||||||
|  |     } else if (_source == "ps/running/list") { | ||||||
|  |         // list of running processes | ||||||
|  |         m_map.insert(_source, "ps"); | ||||||
|  |         m_formatter["ps"] = AWKeysAggregator::FormatterType::List; | ||||||
|  |     } else if (_source == "ps/total/count") { | ||||||
|  |         // total processes count | ||||||
|  |         m_map.insert(_source, "pstot"); | ||||||
|  |         m_formatter["pstot"] = AWKeysAggregator::FormatterType::NoFormat; | ||||||
|  |     } else if (_source.startsWith("quotes")) { | ||||||
|  |         // quotes | ||||||
|  |         QString key = _source; | ||||||
|  |         key.remove("quotes/"); | ||||||
|  |         m_map.insert(_source, key); | ||||||
|  |         m_formatter[key] = AWKeysAggregator::FormatterType::Quotes; | ||||||
|  |     } else if (_source == "mem/swap/free") { | ||||||
|  |         // free swap | ||||||
|  |         // mb | ||||||
|  |         m_map.insert(_source, "swapfreemb"); | ||||||
|  |         m_formatter["swapfreemb"] = AWKeysAggregator::FormatterType::MemMBFormat; | ||||||
|  |         // gb | ||||||
|  |         m_map.insert(_source, "swapfreegb"); | ||||||
|  |         m_formatter["swapfreegb"] = AWKeysAggregator::FormatterType::MemGBFormat; | ||||||
|  |     } else if (_source == "mem/swap/used") { | ||||||
|  |         // used swap | ||||||
|  |         // mb | ||||||
|  |         m_map.insert(_source, "swapmb"); | ||||||
|  |         m_formatter["swapmb"] = AWKeysAggregator::FormatterType::MemMBFormat; | ||||||
|  |         // gb | ||||||
|  |         m_map.insert(_source, "swapgb"); | ||||||
|  |         m_formatter["swapgb"] = AWKeysAggregator::FormatterType::MemGBFormat; | ||||||
|  |     } else if (_source.startsWith("lmsensors/")) { | ||||||
|  |         // temperature | ||||||
|  |         int index = m_devices["temp"].indexOf(_source); | ||||||
|  |         // HACK on DE initialization there are no units key | ||||||
|  |         if (_units.isEmpty()) | ||||||
|  |             return QStringList({QString("temp%1").arg(index)}); | ||||||
|  |         if (index > -1) { | ||||||
|  |             QString key = QString("temp%1").arg(index); | ||||||
|  |             m_map.insert(_source, key); | ||||||
|  |             m_formatter[key] = _units == "°C" ? AWKeysAggregator::FormatterType::Temperature | ||||||
|  |                                               : AWKeysAggregator::FormatterType::Integer; | ||||||
|  |         } | ||||||
|  |     } else if (_source == "Local") { | ||||||
|  |         // time | ||||||
|  |         m_map.insert(_source, "time"); | ||||||
|  |         m_formatter["time"] = AWKeysAggregator::FormatterType::Time; | ||||||
|  |         // custom time | ||||||
|  |         m_map.insert(_source, "ctime"); | ||||||
|  |         m_formatter["ctime"] = AWKeysAggregator::FormatterType::TimeCustom; | ||||||
|  |         // ISO time | ||||||
|  |         m_map.insert(_source, "isotime"); | ||||||
|  |         m_formatter["isotime"] = AWKeysAggregator::FormatterType::TimeISO; | ||||||
|  |         // long time | ||||||
|  |         m_map.insert(_source, "longtime"); | ||||||
|  |         m_formatter["longtime"] = AWKeysAggregator::FormatterType::TimeLong; | ||||||
|  |         // short time | ||||||
|  |         m_map.insert(_source, "shorttime"); | ||||||
|  |         m_formatter["shorttime"] = AWKeysAggregator::FormatterType::TimeShort; | ||||||
|  |         // timestamp | ||||||
|  |         m_map.insert(_source, "tstime"); | ||||||
|  |         m_formatter["tstime"] = AWKeysAggregator::FormatterType::Timestamp; | ||||||
|  |     } else if (_source == "system/brightness") { | ||||||
|  |         m_map.insert(_source, "brightness"); | ||||||
|  |         m_formatter["brightness"] = AWKeysAggregator::FormatterType::IntegerThree; | ||||||
|  |     } else if (_source == "system/volume") { | ||||||
|  |         m_map.insert(_source, "volume"); | ||||||
|  |         m_formatter["volume"] = AWKeysAggregator::FormatterType::IntegerThree; | ||||||
|  |     } else if (_source == "system/uptime") { | ||||||
|  |         // uptime | ||||||
|  |         m_map.insert(_source, "uptime"); | ||||||
|  |         m_formatter["uptime"] = AWKeysAggregator::FormatterType::Uptime; | ||||||
|  |         // custom uptime | ||||||
|  |         m_map.insert(_source, "cuptime"); | ||||||
|  |         m_formatter["cuptime"] = AWKeysAggregator::FormatterType::UptimeCustom; | ||||||
|  |     } else if (_source.startsWith("weather/temperature")) { | ||||||
|  |         // temperature | ||||||
|  |         QString key = _source; | ||||||
|  |         key.remove("weather/"); | ||||||
|  |         m_map.insert(_source, key); | ||||||
|  |         m_formatter[key] = AWKeysAggregator::FormatterType::Temperature; | ||||||
|  |     } else if (_source.startsWith("weather/")) { | ||||||
|  |         // other weather | ||||||
|  |         QString key = _source; | ||||||
|  |         key.remove("weather/"); | ||||||
|  |         m_map.insert(_source, key); | ||||||
|  |         m_formatter[key] = AWKeysAggregator::FormatterType::NoFormat; | ||||||
|  |     } else if (_source.startsWith("load/load")) { | ||||||
|  |         // load source | ||||||
|  |         QString key = _source; | ||||||
|  |         key.remove("load/"); | ||||||
|  |         m_map.insert(_source, key); | ||||||
|  |         m_formatter[key] = AWKeysAggregator::FormatterType::Temperature; | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     QStringList foundKeys = keysFromSource(_source); | ||||||
|  |  | ||||||
|  |     // rewrite formatters for custom ones | ||||||
|  |     QStringList customFormattersKeys; | ||||||
|  |     if (m_customFormatters) | ||||||
|  |         customFormattersKeys = m_customFormatters->definedFormatters(); | ||||||
|  |     qCInfo(LOG_AW) << "Looking for formatters" << foundKeys << "in" << customFormattersKeys; | ||||||
|  |     for (auto &key : foundKeys) { | ||||||
|  |         if (!customFormattersKeys.contains(key)) | ||||||
|  |             continue; | ||||||
|  |         m_formatter[key] = AWKeysAggregator::FormatterType::Custom; | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     // drop key from dictionary if no one user requested key required it | ||||||
|  |     qCInfo(LOG_AW) << "Looking for keys" << foundKeys << "in" << _keys; | ||||||
|  |     bool required = _keys.isEmpty() || std::any_of(foundKeys.cbegin(), foundKeys.cend(), [&_keys](const QString &key) { | ||||||
|  |                         return _keys.contains(key); | ||||||
|  |                     }); | ||||||
|  |     if (!required) { | ||||||
|  |         m_map.remove(_source); | ||||||
|  |         for (auto &key : foundKeys) | ||||||
|  |             m_formatter.remove(key); | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     return keysFromSource(_source); | ||||||
|  | } | ||||||
|  |  | ||||||
|  |  | ||||||
|  | void AWDataEngineMapper::setDevices(const QHash<QString, QStringList> &_devices) | ||||||
|  | { | ||||||
|  |     qCDebug(LOG_AW) << "Devices" << _devices; | ||||||
|  |  | ||||||
|  |     m_devices = _devices; | ||||||
|  | } | ||||||
							
								
								
									
										53
									
								
								sources/awesome-widget/plugin/awdataenginemapper.h
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										53
									
								
								sources/awesome-widget/plugin/awdataenginemapper.h
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,53 @@ | |||||||
|  | /*************************************************************************** | ||||||
|  |  *   This file is part of awesome-widgets                                  * | ||||||
|  |  *                                                                         * | ||||||
|  |  *   awesome-widgets is free software: you can redistribute it and/or      * | ||||||
|  |  *   modify it under the terms of the GNU General Public License as        * | ||||||
|  |  *   published by the Free Software Foundation, either version 3 of the    * | ||||||
|  |  *   License, or (at your option) any later version.                       * | ||||||
|  |  *                                                                         * | ||||||
|  |  *   awesome-widgets is distributed in the hope that it will be useful,    * | ||||||
|  |  *   but WITHOUT ANY WARRANTY; without even the implied warranty of        * | ||||||
|  |  *   MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the         * | ||||||
|  |  *   GNU General Public License for more details.                          * | ||||||
|  |  *                                                                         * | ||||||
|  |  *   You should have received a copy of the GNU General Public License     * | ||||||
|  |  *   along with awesome-widgets. If not, see http://www.gnu.org/licenses/  * | ||||||
|  |  ***************************************************************************/ | ||||||
|  |  | ||||||
|  |  | ||||||
|  | #ifndef AWDATAENGINEMAPPER_H | ||||||
|  | #define AWDATAENGINEMAPPER_H | ||||||
|  |  | ||||||
|  | #include <QMultiHash> | ||||||
|  | #include <QObject> | ||||||
|  |  | ||||||
|  | #include "awkeysaggregator.h" | ||||||
|  |  | ||||||
|  |  | ||||||
|  | class AWFormatterHelper; | ||||||
|  |  | ||||||
|  | class AWDataEngineMapper : public QObject | ||||||
|  | { | ||||||
|  |     Q_OBJECT | ||||||
|  |  | ||||||
|  | public: | ||||||
|  |     explicit AWDataEngineMapper(QObject *_parent = nullptr, AWFormatterHelper *_custom = nullptr); | ||||||
|  |     ~AWDataEngineMapper() override; | ||||||
|  |     // get methods | ||||||
|  |     [[nodiscard]] AWKeysAggregator::FormatterType formatter(const QString &_key) const; | ||||||
|  |     [[nodiscard]] QStringList keysFromSource(const QString &_source) const; | ||||||
|  |     // set methods | ||||||
|  |     QStringList registerSource(const QString &_source, const QString &_units, const QStringList &_keys); | ||||||
|  |     void setDevices(const QHash<QString, QStringList> &_devices); | ||||||
|  |  | ||||||
|  | private: | ||||||
|  |     AWFormatterHelper *m_customFormatters = nullptr; | ||||||
|  |     // variables | ||||||
|  |     QHash<QString, QStringList> m_devices; | ||||||
|  |     QHash<QString, AWKeysAggregator::FormatterType> m_formatter; | ||||||
|  |     QMultiHash<QString, QString> m_map; | ||||||
|  | }; | ||||||
|  |  | ||||||
|  |  | ||||||
|  | #endif /* AWDATAENGINEMAPPER_H */ | ||||||
							
								
								
									
										108
									
								
								sources/awesome-widget/plugin/awdbusadaptor.cpp
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										108
									
								
								sources/awesome-widget/plugin/awdbusadaptor.cpp
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,108 @@ | |||||||
|  | /*************************************************************************** | ||||||
|  |  *   This file is part of awesome-widgets                                  * | ||||||
|  |  *                                                                         * | ||||||
|  |  *   awesome-widgets is free software: you can redistribute it and/or      * | ||||||
|  |  *   modify it under the terms of the GNU General Public License as        * | ||||||
|  |  *   published by the Free Software Foundation, either version 3 of the    * | ||||||
|  |  *   License, or (at your option) any later version.                       * | ||||||
|  |  *                                                                         * | ||||||
|  |  *   awesome-widgets is distributed in the hope that it will be useful,    * | ||||||
|  |  *   but WITHOUT ANY WARRANTY; without even the implied warranty of        * | ||||||
|  |  *   MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the         * | ||||||
|  |  *   GNU General Public License for more details.                          * | ||||||
|  |  *                                                                         * | ||||||
|  |  *   You should have received a copy of the GNU General Public License     * | ||||||
|  |  *   along with awesome-widgets. If not, see http://www.gnu.org/licenses/  * | ||||||
|  |  ***************************************************************************/ | ||||||
|  |  | ||||||
|  | #include "awdbusadaptor.h" | ||||||
|  |  | ||||||
|  | #include <QDBusConnection> | ||||||
|  | #include <QDBusConnectionInterface> | ||||||
|  |  | ||||||
|  | #include "awdebug.h" | ||||||
|  | #include "awkeys.h" | ||||||
|  |  | ||||||
|  |  | ||||||
|  | AWDBusAdaptor::AWDBusAdaptor(AWKeys *_parent) | ||||||
|  |     : QDBusAbstractAdaptor(_parent) | ||||||
|  |     , m_plugin(_parent) | ||||||
|  | { | ||||||
|  |     qCDebug(LOG_DBUS) << __PRETTY_FUNCTION__; | ||||||
|  | } | ||||||
|  |  | ||||||
|  |  | ||||||
|  | AWDBusAdaptor::~AWDBusAdaptor() | ||||||
|  | { | ||||||
|  |     qCDebug(LOG_DBUS) << __PRETTY_FUNCTION__; | ||||||
|  | } | ||||||
|  |  | ||||||
|  |  | ||||||
|  | QStringList AWDBusAdaptor::ActiveServices() | ||||||
|  | { | ||||||
|  |     QDBusMessage listServices = QDBusConnection::sessionBus().interface()->call(QDBus::BlockWithGui, "ListNames"); | ||||||
|  |     if (listServices.arguments().isEmpty()) { | ||||||
|  |         qCWarning(LOG_DBUS) << "Could not find any DBus service"; | ||||||
|  |         return {}; | ||||||
|  |     } | ||||||
|  |     QStringList arguments = listServices.arguments().first().toStringList(); | ||||||
|  |  | ||||||
|  |     return std::accumulate(arguments.cbegin(), arguments.cend(), QStringList(), | ||||||
|  |                            [](QStringList source, const QString &service) { | ||||||
|  |                                if (service.startsWith(AWDBUS_SERVICE)) | ||||||
|  |                                    source.append(service); | ||||||
|  |                                return source; | ||||||
|  |                            }); | ||||||
|  | } | ||||||
|  |  | ||||||
|  |  | ||||||
|  | QString AWDBusAdaptor::Info(const QString &key) const | ||||||
|  | { | ||||||
|  |     return m_plugin->infoByKey(key); | ||||||
|  | } | ||||||
|  |  | ||||||
|  |  | ||||||
|  | QStringList AWDBusAdaptor::Keys(const QString ®exp) const | ||||||
|  | { | ||||||
|  |     return m_plugin->dictKeys(true, regexp); | ||||||
|  | } | ||||||
|  |  | ||||||
|  |  | ||||||
|  | QString AWDBusAdaptor::Value(const QString &key) const | ||||||
|  | { | ||||||
|  |     return m_plugin->valueByKey(key); | ||||||
|  | } | ||||||
|  |  | ||||||
|  |  | ||||||
|  | qlonglong AWDBusAdaptor::WhoAmI() const | ||||||
|  | { | ||||||
|  |     return reinterpret_cast<qlonglong>(m_plugin); | ||||||
|  | } | ||||||
|  |  | ||||||
|  |  | ||||||
|  | void AWDBusAdaptor::SetLogLevel(const QString &what, const int level) | ||||||
|  | { | ||||||
|  |     qCDebug(LOG_DBUS) << "Set log level" << level << "for" << what; | ||||||
|  |  | ||||||
|  |     if (level >= m_logLevels.count()) { | ||||||
|  |         qCDebug(LOG_DBUS) << "Invalid logging level" << level << "should be less than" << m_logLevels.count(); | ||||||
|  |         return; | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     for (auto &lev : m_logLevels) | ||||||
|  |         SetLogLevel(what, lev, m_logLevels.indexOf(lev) >= level); | ||||||
|  | } | ||||||
|  |  | ||||||
|  |  | ||||||
|  | void AWDBusAdaptor::SetLogLevel(const QString &what, const QString &level, const bool enabled) | ||||||
|  | { | ||||||
|  |     qCDebug(LOG_DBUS) << "Set log level" << level << "enabled" << enabled << "for" << what; | ||||||
|  |  | ||||||
|  |     if (!m_logLevels.contains(level)) { | ||||||
|  |         qCDebug(LOG_DBUS) << "Invalid logging level" << level << "should be in" << m_logLevels; | ||||||
|  |         return; | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     QString state = enabled ? "true" : "false"; | ||||||
|  |     QLoggingCategory::setFilterRules(QString("%1.%2=%3").arg(what).arg(level).arg(state)); | ||||||
|  | } | ||||||
							
								
								
									
										55
									
								
								sources/awesome-widget/plugin/awdbusadaptor.h
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										55
									
								
								sources/awesome-widget/plugin/awdbusadaptor.h
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,55 @@ | |||||||
|  | /*************************************************************************** | ||||||
|  |  *   This file is part of awesome-widgets                                  * | ||||||
|  |  *                                                                         * | ||||||
|  |  *   awesome-widgets is free software: you can redistribute it and/or      * | ||||||
|  |  *   modify it under the terms of the GNU General Public License as        * | ||||||
|  |  *   published by the Free Software Foundation, either version 3 of the    * | ||||||
|  |  *   License, or (at your option) any later version.                       * | ||||||
|  |  *                                                                         * | ||||||
|  |  *   awesome-widgets is distributed in the hope that it will be useful,    * | ||||||
|  |  *   but WITHOUT ANY WARRANTY; without even the implied warranty of        * | ||||||
|  |  *   MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the         * | ||||||
|  |  *   GNU General Public License for more details.                          * | ||||||
|  |  *                                                                         * | ||||||
|  |  *   You should have received a copy of the GNU General Public License     * | ||||||
|  |  *   along with awesome-widgets. If not, see http://www.gnu.org/licenses/  * | ||||||
|  |  ***************************************************************************/ | ||||||
|  |  | ||||||
|  |  | ||||||
|  | #ifndef AWDBUSADAPTOR_H | ||||||
|  | #define AWDBUSADAPTOR_H | ||||||
|  |  | ||||||
|  | #include <QDBusAbstractAdaptor> | ||||||
|  |  | ||||||
|  | #include "version.h" | ||||||
|  |  | ||||||
|  |  | ||||||
|  | class AWKeys; | ||||||
|  |  | ||||||
|  | class AWDBusAdaptor : public QDBusAbstractAdaptor | ||||||
|  | { | ||||||
|  |     Q_OBJECT | ||||||
|  |     Q_CLASSINFO("D-Bus Interface", AWDBUS_SERVICE_NAME) | ||||||
|  |  | ||||||
|  | public: | ||||||
|  |     explicit AWDBusAdaptor(AWKeys *_parent = nullptr); | ||||||
|  |     ~AWDBusAdaptor() override; | ||||||
|  |  | ||||||
|  | public slots: | ||||||
|  |     // get methods | ||||||
|  |     [[nodiscard]] static QStringList ActiveServices(); | ||||||
|  |     [[nodiscard]] QString Info(const QString &key) const; | ||||||
|  |     [[nodiscard]] QStringList Keys(const QString ®exp) const; | ||||||
|  |     [[nodiscard]] QString Value(const QString &key) const; | ||||||
|  |     [[nodiscard]] qlonglong WhoAmI() const; | ||||||
|  |     // set methods | ||||||
|  |     void SetLogLevel(const QString &what, int level); | ||||||
|  |     void SetLogLevel(const QString &what, const QString &level, bool enabled); | ||||||
|  |  | ||||||
|  | private: | ||||||
|  |     AWKeys *m_plugin = nullptr; | ||||||
|  |     QStringList m_logLevels = {"debug", "info", "warning", "critical"}; | ||||||
|  | }; | ||||||
|  |  | ||||||
|  |  | ||||||
|  | #endif /* AWDBUSADAPTOR_H */ | ||||||
| @ -20,8 +20,11 @@ | |||||||
| #include <QtQml> | #include <QtQml> | ||||||
|  |  | ||||||
| #include "awactions.h" | #include "awactions.h" | ||||||
|  | #include "awbugreporter.h" | ||||||
| #include "awconfighelper.h" | #include "awconfighelper.h" | ||||||
| #include "awkeys.h" | #include "awkeys.h" | ||||||
|  | #include "awpairconfigfactory.h" | ||||||
|  | #include "awtelemetryhandler.h" | ||||||
|  |  | ||||||
|  |  | ||||||
| void AWPlugin::registerTypes(const char *uri) | void AWPlugin::registerTypes(const char *uri) | ||||||
| @ -29,6 +32,9 @@ void AWPlugin::registerTypes(const char *uri) | |||||||
|     Q_ASSERT(uri == QLatin1String("org.kde.plasma.private.awesomewidget")); |     Q_ASSERT(uri == QLatin1String("org.kde.plasma.private.awesomewidget")); | ||||||
|  |  | ||||||
|     qmlRegisterType<AWActions>(uri, 1, 0, "AWActions"); |     qmlRegisterType<AWActions>(uri, 1, 0, "AWActions"); | ||||||
|  |     qmlRegisterType<AWBugReporter>(uri, 1, 0, "AWBugReporter"); | ||||||
|     qmlRegisterType<AWConfigHelper>(uri, 1, 0, "AWConfigHelper"); |     qmlRegisterType<AWConfigHelper>(uri, 1, 0, "AWConfigHelper"); | ||||||
|  |     qmlRegisterType<AWPairConfigFactory>(uri, 1, 0, "AWPairConfigFactory"); | ||||||
|     qmlRegisterType<AWKeys>(uri, 1, 0, "AWKeys"); |     qmlRegisterType<AWKeys>(uri, 1, 0, "AWKeys"); | ||||||
|  |     qmlRegisterType<AWTelemetryHandler>(uri, 1, 0, "AWTelemetryHandler"); | ||||||
| } | } | ||||||
|  | |||||||
| @ -28,7 +28,7 @@ class AWPlugin : public QQmlExtensionPlugin | |||||||
|     Q_PLUGIN_METADATA(IID "org.qt-project.Qt.QQmlExtensionInterface") |     Q_PLUGIN_METADATA(IID "org.qt-project.Qt.QQmlExtensionInterface") | ||||||
|  |  | ||||||
| public: | public: | ||||||
|     void registerTypes(const char *uri); |     void registerTypes(const char *uri) override; | ||||||
| }; | }; | ||||||
|  |  | ||||||
|  |  | ||||||
|  | |||||||
							
								
								
									
										37
									
								
								sources/awesome-widget/plugin/awformatterconfig.cpp
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										37
									
								
								sources/awesome-widget/plugin/awformatterconfig.cpp
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,37 @@ | |||||||
|  | /*************************************************************************** | ||||||
|  |  *   This file is part of awesome-widgets                                  * | ||||||
|  |  *                                                                         * | ||||||
|  |  *   awesome-widgets is free software: you can redistribute it and/or      * | ||||||
|  |  *   modify it under the terms of the GNU General Public License as        * | ||||||
|  |  *   published by the Free Software Foundation, either version 3 of the    * | ||||||
|  |  *   License, or (at your option) any later version.                       * | ||||||
|  |  *                                                                         * | ||||||
|  |  *   awesome-widgets is distributed in the hope that it will be useful,    * | ||||||
|  |  *   but WITHOUT ANY WARRANTY; without even the implied warranty of        * | ||||||
|  |  *   MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the         * | ||||||
|  |  *   GNU General Public License for more details.                          * | ||||||
|  |  *                                                                         * | ||||||
|  |  *   You should have received a copy of the GNU General Public License     * | ||||||
|  |  *   along with awesome-widgets. If not, see http://www.gnu.org/licenses/  * | ||||||
|  |  ***************************************************************************/ | ||||||
|  |  | ||||||
|  | #include "awformatterconfig.h" | ||||||
|  |  | ||||||
|  | #include "awdebug.h" | ||||||
|  | #include "awformatterhelper.h" | ||||||
|  |  | ||||||
|  |  | ||||||
|  | AWFormatterConfig::AWFormatterConfig(QWidget *_parent, const QStringList &_keys) | ||||||
|  |     : AWAbstractPairConfig(_parent, true, _keys) | ||||||
|  | { | ||||||
|  |     qCDebug(LOG_AW) << __PRETTY_FUNCTION__; | ||||||
|  |  | ||||||
|  |     setEditable(false, false); | ||||||
|  |     initHelper<AWFormatterHelper>(); | ||||||
|  | } | ||||||
|  |  | ||||||
|  |  | ||||||
|  | AWFormatterConfig::~AWFormatterConfig() | ||||||
|  | { | ||||||
|  |     qCDebug(LOG_AW) << __PRETTY_FUNCTION__; | ||||||
|  | } | ||||||
							
								
								
									
										35
									
								
								sources/awesome-widget/plugin/awformatterconfig.h
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										35
									
								
								sources/awesome-widget/plugin/awformatterconfig.h
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,35 @@ | |||||||
|  | /*************************************************************************** | ||||||
|  |  *   This file is part of awesome-widgets                                  * | ||||||
|  |  *                                                                         * | ||||||
|  |  *   awesome-widgets is free software: you can redistribute it and/or      * | ||||||
|  |  *   modify it under the terms of the GNU General Public License as        * | ||||||
|  |  *   published by the Free Software Foundation, either version 3 of the    * | ||||||
|  |  *   License, or (at your option) any later version.                       * | ||||||
|  |  *                                                                         * | ||||||
|  |  *   awesome-widgets is distributed in the hope that it will be useful,    * | ||||||
|  |  *   but WITHOUT ANY WARRANTY; without even the implied warranty of        * | ||||||
|  |  *   MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the         * | ||||||
|  |  *   GNU General Public License for more details.                          * | ||||||
|  |  *                                                                         * | ||||||
|  |  *   You should have received a copy of the GNU General Public License     * | ||||||
|  |  *   along with awesome-widgets. If not, see http://www.gnu.org/licenses/  * | ||||||
|  |  ***************************************************************************/ | ||||||
|  |  | ||||||
|  |  | ||||||
|  | #ifndef AWFORMATTERCONFIG_H | ||||||
|  | #define AWFORMATTERCONFIG_H | ||||||
|  |  | ||||||
|  | #include "awabstractpairconfig.h" | ||||||
|  |  | ||||||
|  |  | ||||||
|  | class AWFormatterConfig : public AWAbstractPairConfig | ||||||
|  | { | ||||||
|  |     Q_OBJECT | ||||||
|  |  | ||||||
|  | public: | ||||||
|  |     explicit AWFormatterConfig(QWidget *_parent = nullptr, const QStringList &_keys = QStringList()); | ||||||
|  |     ~AWFormatterConfig() override; | ||||||
|  | }; | ||||||
|  |  | ||||||
|  |  | ||||||
|  | #endif /* AWFORMATTERCONFIG_H */ | ||||||
							
								
								
									
										240
									
								
								sources/awesome-widget/plugin/awformatterhelper.cpp
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										240
									
								
								sources/awesome-widget/plugin/awformatterhelper.cpp
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,240 @@ | |||||||
|  | /*************************************************************************** | ||||||
|  |  *   This file is part of awesome-widgets                                  * | ||||||
|  |  *                                                                         * | ||||||
|  |  *   awesome-widgets is free software: you can redistribute it and/or      * | ||||||
|  |  *   modify it under the terms of the GNU General Public License as        * | ||||||
|  |  *   published by the Free Software Foundation, either version 3 of the    * | ||||||
|  |  *   License, or (at your option) any later version.                       * | ||||||
|  |  *                                                                         * | ||||||
|  |  *   awesome-widgets is distributed in the hope that it will be useful,    * | ||||||
|  |  *   but WITHOUT ANY WARRANTY; without even the implied warranty of        * | ||||||
|  |  *   MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the         * | ||||||
|  |  *   GNU General Public License for more details.                          * | ||||||
|  |  *                                                                         * | ||||||
|  |  *   You should have received a copy of the GNU General Public License     * | ||||||
|  |  *   along with awesome-widgets. If not, see http://www.gnu.org/licenses/  * | ||||||
|  |  ***************************************************************************/ | ||||||
|  |  | ||||||
|  | #include "awformatterhelper.h" | ||||||
|  |  | ||||||
|  | #include <KI18n/KLocalizedString> | ||||||
|  |  | ||||||
|  | #include <QDir> | ||||||
|  | #include <QInputDialog> | ||||||
|  | #include <QSettings> | ||||||
|  |  | ||||||
|  | #include "awdatetimeformatter.h" | ||||||
|  | #include "awdebug.h" | ||||||
|  | #include "awfloatformatter.h" | ||||||
|  | #include "awjsonformatter.h" | ||||||
|  | #include "awlistformatter.h" | ||||||
|  | #include "awnoformatter.h" | ||||||
|  | #include "awscriptformatter.h" | ||||||
|  | #include "awstringformatter.h" | ||||||
|  |  | ||||||
|  |  | ||||||
|  | AWFormatterHelper::AWFormatterHelper(QWidget *_parent) | ||||||
|  |     : AbstractExtItemAggregator(_parent, "formatters") | ||||||
|  |     , AWAbstractPairHelper("awesomewidgets/formatters/formatters.ini", "Formatters") | ||||||
|  | { | ||||||
|  |     qCDebug(LOG_AW) << __PRETTY_FUNCTION__; | ||||||
|  |  | ||||||
|  |     AWFormatterHelper::initItems(); | ||||||
|  | } | ||||||
|  |  | ||||||
|  |  | ||||||
|  | AWFormatterHelper::~AWFormatterHelper() | ||||||
|  | { | ||||||
|  |     qCDebug(LOG_AW) << __PRETTY_FUNCTION__; | ||||||
|  |  | ||||||
|  |     m_formatters.clear(); | ||||||
|  |     m_formattersClasses.clear(); | ||||||
|  | } | ||||||
|  |  | ||||||
|  |  | ||||||
|  | void AWFormatterHelper::initItems() | ||||||
|  | { | ||||||
|  |     initFormatters(); | ||||||
|  |  | ||||||
|  |     // assign internal storage | ||||||
|  |     m_formatters.clear(); | ||||||
|  |     for (auto &key : pairs().keys()) { | ||||||
|  |         auto name = pairs()[key]; | ||||||
|  |         if (!m_formattersClasses.contains(name)) { | ||||||
|  |             qCWarning(LOG_AW) << "Invalid formatter" << name << "found in" << key; | ||||||
|  |             continue; | ||||||
|  |         } | ||||||
|  |  | ||||||
|  |         m_formatters[key] = m_formattersClasses[name]; | ||||||
|  |     } | ||||||
|  | } | ||||||
|  |  | ||||||
|  |  | ||||||
|  | QString AWFormatterHelper::convert(const QVariant &_value, const QString &_name) const | ||||||
|  | { | ||||||
|  |     qCDebug(LOG_AW) << "Convert value" << _value << "for" << _name; | ||||||
|  |  | ||||||
|  |     return m_formatters.contains(_name) ? m_formatters[_name]->convert(_value) : _value.toString(); | ||||||
|  | } | ||||||
|  |  | ||||||
|  |  | ||||||
|  | QStringList AWFormatterHelper::definedFormatters() const | ||||||
|  | { | ||||||
|  |     return m_formatters.keys(); | ||||||
|  | } | ||||||
|  |  | ||||||
|  |  | ||||||
|  | QList<AbstractExtItem *> AWFormatterHelper::items() const | ||||||
|  | { | ||||||
|  |     QList<AbstractExtItem *> converted; | ||||||
|  |     for (auto &item : m_formattersClasses.values()) | ||||||
|  |         converted.append(item); | ||||||
|  |  | ||||||
|  |     return converted; | ||||||
|  | } | ||||||
|  |  | ||||||
|  |  | ||||||
|  | void AWFormatterHelper::editPairs() | ||||||
|  | { | ||||||
|  |     return editItems(); | ||||||
|  | } | ||||||
|  |  | ||||||
|  |  | ||||||
|  | QStringList AWFormatterHelper::leftKeys() | ||||||
|  | { | ||||||
|  |     return {}; | ||||||
|  | } | ||||||
|  |  | ||||||
|  |  | ||||||
|  | QStringList AWFormatterHelper::rightKeys() | ||||||
|  | { | ||||||
|  |     return m_formattersClasses.keys(); | ||||||
|  | } | ||||||
|  |  | ||||||
|  |  | ||||||
|  | void AWFormatterHelper::editItems() | ||||||
|  | { | ||||||
|  |     repaintList(); | ||||||
|  |     int ret = exec(); | ||||||
|  |     qCInfo(LOG_AW) << "Dialog returns" << ret; | ||||||
|  | } | ||||||
|  |  | ||||||
|  |  | ||||||
|  | AWAbstractFormatter::FormatterClass AWFormatterHelper::defineFormatterClass(const QString &_stringType) | ||||||
|  | { | ||||||
|  |     qCDebug(LOG_AW) << "Define formatter class for" << _stringType; | ||||||
|  |  | ||||||
|  |     AWAbstractFormatter::FormatterClass formatter = AWAbstractFormatter::FormatterClass::NoFormat; | ||||||
|  |     if (_stringType == "DateTime") | ||||||
|  |         formatter = AWAbstractFormatter::FormatterClass::DateTime; | ||||||
|  |     else if (_stringType == "Float") | ||||||
|  |         formatter = AWAbstractFormatter::FormatterClass::Float; | ||||||
|  |     else if (_stringType == "List") | ||||||
|  |         formatter = AWAbstractFormatter::FormatterClass::List; | ||||||
|  |     else if (_stringType == "NoFormat") | ||||||
|  |         ; | ||||||
|  |     else if (_stringType == "Script") | ||||||
|  |         formatter = AWAbstractFormatter::FormatterClass::Script; | ||||||
|  |     else if (_stringType == "String") | ||||||
|  |         formatter = AWAbstractFormatter::FormatterClass::String; | ||||||
|  |     else if (_stringType == "Json") | ||||||
|  |         formatter = AWAbstractFormatter::FormatterClass::Json; | ||||||
|  |     else | ||||||
|  |         qCWarning(LOG_AW) << "Unknown formatter" << _stringType; | ||||||
|  |  | ||||||
|  |     return formatter; | ||||||
|  | } | ||||||
|  |  | ||||||
|  |  | ||||||
|  | void AWFormatterHelper::initFormatters() | ||||||
|  | { | ||||||
|  |     m_formattersClasses.clear(); | ||||||
|  |  | ||||||
|  |     auto dirs = directories(); | ||||||
|  |     for (auto &dir : dirs) { | ||||||
|  |         QStringList files = QDir(dir).entryList(QDir::Files, QDir::Name); | ||||||
|  |         for (auto &file : files) { | ||||||
|  |             // check filename | ||||||
|  |             if (!file.endsWith(".desktop")) | ||||||
|  |                 continue; | ||||||
|  |             qCInfo(LOG_AW) << "Found file" << file << "in" << dir; | ||||||
|  |             QString filePath = QString("%1/%2").arg(dir).arg(file); | ||||||
|  |             // check if already exists | ||||||
|  |             auto values = m_formattersClasses.values(); | ||||||
|  |             if (std::any_of(values.cbegin(), values.cend(), | ||||||
|  |                             [&filePath](const AWAbstractFormatter *item) { return (item->fileName() == filePath); })) | ||||||
|  |                 continue; | ||||||
|  |  | ||||||
|  |             auto metadata = readMetadata(filePath); | ||||||
|  |             switch (metadata.second) { | ||||||
|  |             case AWAbstractFormatter::FormatterClass::DateTime: | ||||||
|  |                 m_formattersClasses[metadata.first] = new AWDateTimeFormatter(this, filePath); | ||||||
|  |                 break; | ||||||
|  |             case AWAbstractFormatter::FormatterClass::Float: | ||||||
|  |                 m_formattersClasses[metadata.first] = new AWFloatFormatter(this, filePath); | ||||||
|  |                 break; | ||||||
|  |             case AWAbstractFormatter::FormatterClass::List: | ||||||
|  |                 m_formattersClasses[metadata.first] = new AWListFormatter(this, filePath); | ||||||
|  |                 break; | ||||||
|  |             case AWAbstractFormatter::FormatterClass::Script: | ||||||
|  |                 m_formattersClasses[metadata.first] = new AWScriptFormatter(this, filePath); | ||||||
|  |                 break; | ||||||
|  |             case AWAbstractFormatter::FormatterClass::String: | ||||||
|  |                 m_formattersClasses[metadata.first] = new AWStringFormatter(this, filePath); | ||||||
|  |                 break; | ||||||
|  |             case AWAbstractFormatter::FormatterClass::Json: | ||||||
|  |                 m_formattersClasses[metadata.first] = new AWJsonFormatter(this, filePath); | ||||||
|  |                 break; | ||||||
|  |             case AWAbstractFormatter::FormatterClass::NoFormat: | ||||||
|  |                 m_formattersClasses[metadata.first] = new AWNoFormatter(this, filePath); | ||||||
|  |                 break; | ||||||
|  |             } | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  | } | ||||||
|  |  | ||||||
|  |  | ||||||
|  | QPair<QString, AWAbstractFormatter::FormatterClass> AWFormatterHelper::readMetadata(const QString &_filePath) | ||||||
|  | { | ||||||
|  |     qCDebug(LOG_AW) << "Read initial parameters from" << _filePath; | ||||||
|  |  | ||||||
|  |     QSettings settings(_filePath, QSettings::IniFormat); | ||||||
|  |     settings.beginGroup("Desktop Entry"); | ||||||
|  |     QString name = settings.value("Name", _filePath).toString(); | ||||||
|  |     QString type = settings.value("X-AW-Type", "NoFormat").toString(); | ||||||
|  |     AWAbstractFormatter::FormatterClass formatter = defineFormatterClass(type); | ||||||
|  |     settings.endGroup(); | ||||||
|  |  | ||||||
|  |     return QPair<QString, AWAbstractFormatter::FormatterClass>(name, formatter); | ||||||
|  | } | ||||||
|  |  | ||||||
|  |  | ||||||
|  | void AWFormatterHelper::doCreateItem() | ||||||
|  | { | ||||||
|  |     QStringList selection = {"NoFormat", "DateTime", "Float", "List", "Script", "String", "Json"}; | ||||||
|  |     bool ok; | ||||||
|  |     QString select = QInputDialog::getItem(this, i18n("Select type"), i18n("Type:"), selection, 0, false, &ok); | ||||||
|  |     if (!ok) { | ||||||
|  |         qCWarning(LOG_AW) << "No type selected"; | ||||||
|  |         return; | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     qCInfo(LOG_AW) << "Selected type" << select; | ||||||
|  |     AWAbstractFormatter::FormatterClass formatter = defineFormatterClass(select); | ||||||
|  |     switch (formatter) { | ||||||
|  |     case AWAbstractFormatter::FormatterClass::DateTime: | ||||||
|  |         return createItem<AWDateTimeFormatter>(); | ||||||
|  |     case AWAbstractFormatter::FormatterClass::Float: | ||||||
|  |         return createItem<AWFloatFormatter>(); | ||||||
|  |     case AWAbstractFormatter::FormatterClass::List: | ||||||
|  |         return createItem<AWListFormatter>(); | ||||||
|  |     case AWAbstractFormatter::FormatterClass::Script: | ||||||
|  |         return createItem<AWScriptFormatter>(); | ||||||
|  |     case AWAbstractFormatter::FormatterClass::String: | ||||||
|  |         return createItem<AWStringFormatter>(); | ||||||
|  |     case AWAbstractFormatter::FormatterClass::Json: | ||||||
|  |         return createItem<AWJsonFormatter>(); | ||||||
|  |     case AWAbstractFormatter::FormatterClass::NoFormat: | ||||||
|  |         return createItem<AWNoFormatter>(); | ||||||
|  |     } | ||||||
|  | } | ||||||
							
								
								
									
										61
									
								
								sources/awesome-widget/plugin/awformatterhelper.h
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										61
									
								
								sources/awesome-widget/plugin/awformatterhelper.h
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,61 @@ | |||||||
|  | /*************************************************************************** | ||||||
|  |  *   This file is part of awesome-widgets                                  * | ||||||
|  |  *                                                                         * | ||||||
|  |  *   awesome-widgets is free software: you can redistribute it and/or      * | ||||||
|  |  *   modify it under the terms of the GNU General Public License as        * | ||||||
|  |  *   published by the Free Software Foundation, either version 3 of the    * | ||||||
|  |  *   License, or (at your option) any later version.                       * | ||||||
|  |  *                                                                         * | ||||||
|  |  *   awesome-widgets is distributed in the hope that it will be useful,    * | ||||||
|  |  *   but WITHOUT ANY WARRANTY; without even the implied warranty of        * | ||||||
|  |  *   MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the         * | ||||||
|  |  *   GNU General Public License for more details.                          * | ||||||
|  |  *                                                                         * | ||||||
|  |  *   You should have received a copy of the GNU General Public License     * | ||||||
|  |  *   along with awesome-widgets. If not, see http://www.gnu.org/licenses/  * | ||||||
|  |  ***************************************************************************/ | ||||||
|  |  | ||||||
|  |  | ||||||
|  | #ifndef AWFORMATTERHELPER_H | ||||||
|  | #define AWFORMATTERHELPER_H | ||||||
|  |  | ||||||
|  | #include "abstractextitemaggregator.h" | ||||||
|  | #include "awabstractformatter.h" | ||||||
|  | #include "awabstractpairhelper.h" | ||||||
|  |  | ||||||
|  |  | ||||||
|  | class AWFormatterHelper : public AbstractExtItemAggregator, public AWAbstractPairHelper | ||||||
|  | { | ||||||
|  |     Q_OBJECT | ||||||
|  |  | ||||||
|  | public: | ||||||
|  |     explicit AWFormatterHelper(QWidget *_parent = nullptr); | ||||||
|  |     ~AWFormatterHelper() override; | ||||||
|  |     // read-write methods | ||||||
|  |     void initItems() override; | ||||||
|  |     // methods | ||||||
|  |     [[nodiscard]] QString convert(const QVariant &_value, const QString &_name) const; | ||||||
|  |     [[nodiscard]] QStringList definedFormatters() const; | ||||||
|  |     [[nodiscard]] QList<AbstractExtItem *> items() const override; | ||||||
|  |     // configuration related | ||||||
|  |     void editPairs() override; | ||||||
|  |     QStringList leftKeys() override; | ||||||
|  |     QStringList rightKeys() override; | ||||||
|  |  | ||||||
|  | public slots: | ||||||
|  |     void editItems(); | ||||||
|  |  | ||||||
|  | private: | ||||||
|  |     // methods | ||||||
|  |     static AWAbstractFormatter::FormatterClass defineFormatterClass(const QString &_stringType); | ||||||
|  |     void initFormatters(); | ||||||
|  |     [[nodiscard]] static QPair<QString, AWAbstractFormatter::FormatterClass> readMetadata(const QString &_filePath); | ||||||
|  |     // parent methods | ||||||
|  |     void doCreateItem() override; | ||||||
|  |     // properties | ||||||
|  |     QHash<QString, AWAbstractFormatter *> m_formatters; | ||||||
|  |     QHash<QString, AWAbstractFormatter *> m_formattersClasses; | ||||||
|  | }; | ||||||
|  |  | ||||||
|  |  | ||||||
|  | #endif /* AWFORMATTERHELPER_H */ | ||||||
							
								
								
									
										174
									
								
								sources/awesome-widget/plugin/awkeycache.cpp
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										174
									
								
								sources/awesome-widget/plugin/awkeycache.cpp
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,174 @@ | |||||||
|  | /*************************************************************************** | ||||||
|  |  *   This file is part of awesome-widgets                                  * | ||||||
|  |  *                                                                         * | ||||||
|  |  *   awesome-widgets is free software: you can redistribute it and/or      * | ||||||
|  |  *   modify it under the terms of the GNU General Public License as        * | ||||||
|  |  *   published by the Free Software Foundation, either version 3 of the    * | ||||||
|  |  *   License, or (at your option) any later version.                       * | ||||||
|  |  *                                                                         * | ||||||
|  |  *   awesome-widgets is distributed in the hope that it will be useful,    * | ||||||
|  |  *   but WITHOUT ANY WARRANTY; without even the implied warranty of        * | ||||||
|  |  *   MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the         * | ||||||
|  |  *   GNU General Public License for more details.                          * | ||||||
|  |  *                                                                         * | ||||||
|  |  *   You should have received a copy of the GNU General Public License     * | ||||||
|  |  *   along with awesome-widgets. If not, see http://www.gnu.org/licenses/  * | ||||||
|  |  ***************************************************************************/ | ||||||
|  |  | ||||||
|  |  | ||||||
|  | #include "awkeycache.h" | ||||||
|  |  | ||||||
|  | #include <QDir> | ||||||
|  | #include <QNetworkInterface> | ||||||
|  | #include <QSettings> | ||||||
|  | #include <QStandardPaths> | ||||||
|  |  | ||||||
|  | #include "awdebug.h" | ||||||
|  |  | ||||||
|  |  | ||||||
|  | bool AWKeyCache::addKeyToCache(const QString &_type, const QString &_key) | ||||||
|  | { | ||||||
|  |     qCDebug(LOG_AW) << "Key" << _key << "with type" << _type; | ||||||
|  |  | ||||||
|  |     QString fileName | ||||||
|  |         = QString("%1/awesomewidgets.ndx").arg(QStandardPaths::writableLocation(QStandardPaths::GenericCacheLocation)); | ||||||
|  |     qCInfo(LOG_AW) << "Cache file" << fileName; | ||||||
|  |     QSettings cache(fileName, QSettings::IniFormat); | ||||||
|  |  | ||||||
|  |     cache.beginGroup(_type); | ||||||
|  |     QStringList cachedValues; | ||||||
|  |     for (auto &number : cache.allKeys()) | ||||||
|  |         cachedValues.append(cache.value(number).toString()); | ||||||
|  |  | ||||||
|  |     if (_type == "hdd") { | ||||||
|  |         QStringList allDevices = QDir("/dev").entryList(QDir::System, QDir::Name); | ||||||
|  |         QStringList devices = allDevices.filter(QRegExp("^[hms]d[a-z]$")); | ||||||
|  |         for (auto &dev : devices) { | ||||||
|  |             QString device = QString("/dev/%1").arg(dev); | ||||||
|  |             if (cachedValues.contains(device)) | ||||||
|  |                 continue; | ||||||
|  |             qCInfo(LOG_AW) << "Found new key" << device << "for type" << _type; | ||||||
|  |             cachedValues.append(device); | ||||||
|  |             cache.setValue(QString("%1").arg(cache.allKeys().count(), 3, 10, QChar('0')), device); | ||||||
|  |         } | ||||||
|  |     } else if (_type == "net") { | ||||||
|  |         QList<QNetworkInterface> rawInterfaceList = QNetworkInterface::allInterfaces(); | ||||||
|  |         for (auto &interface : rawInterfaceList) { | ||||||
|  |             QString device = interface.name(); | ||||||
|  |             if (cachedValues.contains(device)) | ||||||
|  |                 continue; | ||||||
|  |             qCInfo(LOG_AW) << "Found new key" << device << "for type" << _type; | ||||||
|  |             cachedValues.append(device); | ||||||
|  |             cache.setValue(QString("%1").arg(cache.allKeys().count(), 3, 10, QChar('0')), device); | ||||||
|  |         } | ||||||
|  |     } else { | ||||||
|  |         if (cachedValues.contains(_key)) | ||||||
|  |             return false; | ||||||
|  |         qCInfo(LOG_AW) << "Found new key" << _key << "for type" << _type; | ||||||
|  |         cache.setValue(QString("%1").arg(cache.allKeys().count(), 3, 10, QChar('0')), _key); | ||||||
|  |     } | ||||||
|  |     cache.endGroup(); | ||||||
|  |  | ||||||
|  |     cache.sync(); | ||||||
|  |     return true; | ||||||
|  | } | ||||||
|  |  | ||||||
|  |  | ||||||
|  | QStringList AWKeyCache::getRequiredKeys(const QStringList &_keys, const QStringList &_bars, const QVariantMap &_tooltip, | ||||||
|  |                                         const QStringList &_userKeys, const QStringList &_allKeys) | ||||||
|  | { | ||||||
|  |     qCDebug(LOG_AW) << "Looking for required keys in" << _keys << _bars << "using tooltip settings" << _tooltip; | ||||||
|  |  | ||||||
|  |     // initial copy | ||||||
|  |     QSet<QString> used(_keys.cbegin(), _keys.cend()); | ||||||
|  |     used.unite(QSet(_bars.cbegin(), _bars.cend())); | ||||||
|  |     used.unite(QSet(_userKeys.cbegin(), _userKeys.cend())); | ||||||
|  |     // insert keys from tooltip | ||||||
|  |     for (auto &key : _tooltip.keys()) { | ||||||
|  |         if ((key.endsWith("Tooltip")) && (_tooltip[key].toBool())) { | ||||||
|  |             key.remove("Tooltip"); | ||||||
|  |             used << key; | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     // insert depending keys, refer to AWKeys::calculateValues() | ||||||
|  |     // hddtotmb* | ||||||
|  |     for (auto &key : _allKeys.filter(QRegExp("^hddtotmb"))) { | ||||||
|  |         if (!used.contains(key)) | ||||||
|  |             continue; | ||||||
|  |         key.remove("hddtotmb"); | ||||||
|  |         int index = key.toInt(); | ||||||
|  |         used << QString("hddfreemb%1").arg(index) << QString("hddmb%1").arg(index); | ||||||
|  |     } | ||||||
|  |     // hddtotgb* | ||||||
|  |     for (auto &key : _allKeys.filter(QRegExp("^hddtotgb"))) { | ||||||
|  |         if (!used.contains(key)) | ||||||
|  |             continue; | ||||||
|  |         key.remove("hddtotgb"); | ||||||
|  |         int index = key.toInt(); | ||||||
|  |         used << QString("hddfreegb%1").arg(index) << QString("hddgb%1").arg(index); | ||||||
|  |     } | ||||||
|  |     // mem | ||||||
|  |     if (used.contains("mem")) | ||||||
|  |         used << "memmb" | ||||||
|  |              << "memtotmb"; | ||||||
|  |     // memtotmb | ||||||
|  |     if (used.contains("memtotmb")) | ||||||
|  |         used << "memusedmb" | ||||||
|  |              << "memfreemb"; | ||||||
|  |     // memtotgb | ||||||
|  |     if (used.contains("memtotgb")) | ||||||
|  |         used << "memusedgb" | ||||||
|  |              << "memfreegb"; | ||||||
|  |     // swap | ||||||
|  |     if (used.contains("swap")) | ||||||
|  |         used << "swapmb" | ||||||
|  |              << "swaptotmb"; | ||||||
|  |     // swaptotmb | ||||||
|  |     if (used.contains("swaptotmb")) | ||||||
|  |         used << "swapmb" | ||||||
|  |              << "swapfreemb"; | ||||||
|  |     // memtotgb | ||||||
|  |     if (used.contains("swaptotgb")) | ||||||
|  |         used << "swapgb" | ||||||
|  |              << "swapfreegb"; | ||||||
|  |     // network keys | ||||||
|  |     QStringList netKeys( | ||||||
|  |         {"up", "upkb", "uptot", "uptotkb", "upunits", "down", "downkb", "downtot", "downtotkb", "downunits"}); | ||||||
|  |     for (auto &key : netKeys) { | ||||||
|  |         if (!used.contains(key)) | ||||||
|  |             continue; | ||||||
|  |         QStringList filt = _allKeys.filter(QRegExp(QString("^%1[0-9]{1,}").arg(key))); | ||||||
|  |         for (auto &filtered : filt) | ||||||
|  |             used << filtered; | ||||||
|  |     } | ||||||
|  |     // netdev key | ||||||
|  |     if (std::any_of(netKeys.cbegin(), netKeys.cend(), [&used](const QString &key) { return used.contains(key); })) | ||||||
|  |         used << "netdev"; | ||||||
|  |  | ||||||
|  |     // HACK append dummy if there are no other keys. This hack is required | ||||||
|  |     // because empty list leads to the same behaviour as skip checking | ||||||
|  |     if (used.isEmpty()) | ||||||
|  |         used << "dummy"; | ||||||
|  |  | ||||||
|  |     return used.values(); | ||||||
|  | } | ||||||
|  |  | ||||||
|  |  | ||||||
|  | QHash<QString, QStringList> AWKeyCache::loadKeysFromCache() | ||||||
|  | { | ||||||
|  |     QString fileName | ||||||
|  |         = QString("%1/awesomewidgets.ndx").arg(QStandardPaths::writableLocation(QStandardPaths::GenericCacheLocation)); | ||||||
|  |     qCInfo(LOG_AW) << "Cache file" << fileName; | ||||||
|  |     QSettings cache(fileName, QSettings::IniFormat); | ||||||
|  |  | ||||||
|  |     QHash<QString, QStringList> devices; | ||||||
|  |     for (auto &group : cache.childGroups()) { | ||||||
|  |         cache.beginGroup(group); | ||||||
|  |         for (auto &key : cache.allKeys()) | ||||||
|  |             devices[group].append(cache.value(key).toString()); | ||||||
|  |         cache.endGroup(); | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     return devices; | ||||||
|  | } | ||||||
							
								
								
									
										36
									
								
								sources/awesome-widget/plugin/awkeycache.h
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										36
									
								
								sources/awesome-widget/plugin/awkeycache.h
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,36 @@ | |||||||
|  | /*************************************************************************** | ||||||
|  |  *   This file is part of awesome-widgets                                  * | ||||||
|  |  *                                                                         * | ||||||
|  |  *   awesome-widgets is free software: you can redistribute it and/or      * | ||||||
|  |  *   modify it under the terms of the GNU General Public License as        * | ||||||
|  |  *   published by the Free Software Foundation, either version 3 of the    * | ||||||
|  |  *   License, or (at your option) any later version.                       * | ||||||
|  |  *                                                                         * | ||||||
|  |  *   awesome-widgets is distributed in the hope that it will be useful,    * | ||||||
|  |  *   but WITHOUT ANY WARRANTY; without even the implied warranty of        * | ||||||
|  |  *   MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the         * | ||||||
|  |  *   GNU General Public License for more details.                          * | ||||||
|  |  *                                                                         * | ||||||
|  |  *   You should have received a copy of the GNU General Public License     * | ||||||
|  |  *   along with awesome-widgets. If not, see http://www.gnu.org/licenses/  * | ||||||
|  |  ***************************************************************************/ | ||||||
|  |  | ||||||
|  |  | ||||||
|  | #ifndef AWKEYCACHE_H | ||||||
|  | #define AWKEYCACHE_H | ||||||
|  |  | ||||||
|  | #include <QHash> | ||||||
|  | #include <QString> | ||||||
|  | #include <QVariant> | ||||||
|  |  | ||||||
|  |  | ||||||
|  | namespace AWKeyCache | ||||||
|  | { | ||||||
|  | bool addKeyToCache(const QString &_type, const QString &_key = ""); | ||||||
|  | QStringList getRequiredKeys(const QStringList &_keys, const QStringList &_bars, const QVariantMap &_tooltip, | ||||||
|  |                             const QStringList &_userKeys, const QStringList &_allKeys); | ||||||
|  | QHash<QString, QStringList> loadKeysFromCache(); | ||||||
|  | } // namespace AWKeyCache | ||||||
|  |  | ||||||
|  |  | ||||||
|  | #endif /* AWKEYCACHE_H */ | ||||||
							
								
								
									
										356
									
								
								sources/awesome-widget/plugin/awkeyoperations.cpp
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										356
									
								
								sources/awesome-widget/plugin/awkeyoperations.cpp
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,356 @@ | |||||||
|  | /*************************************************************************** | ||||||
|  |  *   This file is part of awesome-widgets                                  * | ||||||
|  |  *                                                                         * | ||||||
|  |  *   awesome-widgets is free software: you can redistribute it and/or      * | ||||||
|  |  *   modify it under the terms of the GNU General Public License as        * | ||||||
|  |  *   published by the Free Software Foundation, either version 3 of the    * | ||||||
|  |  *   License, or (at your option) any later version.                       * | ||||||
|  |  *                                                                         * | ||||||
|  |  *   awesome-widgets is distributed in the hope that it will be useful,    * | ||||||
|  |  *   but WITHOUT ANY WARRANTY; without even the implied warranty of        * | ||||||
|  |  *   MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the         * | ||||||
|  |  *   GNU General Public License for more details.                          * | ||||||
|  |  *                                                                         * | ||||||
|  |  *   You should have received a copy of the GNU General Public License     * | ||||||
|  |  *   along with awesome-widgets. If not, see http://www.gnu.org/licenses/  * | ||||||
|  |  ***************************************************************************/ | ||||||
|  |  | ||||||
|  | #include "awkeyoperations.h" | ||||||
|  |  | ||||||
|  | #include <QDir> | ||||||
|  | #include <QRegExp> | ||||||
|  | #include <QThread> | ||||||
|  |  | ||||||
|  | #include "awcustomkeyshelper.h" | ||||||
|  | #include "awdebug.h" | ||||||
|  | #include "awkeycache.h" | ||||||
|  | #include "awpatternfunctions.h" | ||||||
|  | // extensions | ||||||
|  | #include "extnetworkrequest.h" | ||||||
|  | #include "extquotes.h" | ||||||
|  | #include "extscript.h" | ||||||
|  | #include "extupgrade.h" | ||||||
|  | #include "extweather.h" | ||||||
|  | #include "graphicalitem.h" | ||||||
|  |  | ||||||
|  |  | ||||||
|  | AWKeyOperations::AWKeyOperations(QObject *_parent) | ||||||
|  |     : QObject(_parent) | ||||||
|  | { | ||||||
|  |     qCDebug(LOG_AW) << __PRETTY_FUNCTION__; | ||||||
|  |  | ||||||
|  |     m_customKeys = new AWCustomKeysHelper(this); | ||||||
|  |     m_graphicalItems = new ExtItemAggregator<GraphicalItem>(nullptr, "desktops"); | ||||||
|  |     m_extNetRequest = new ExtItemAggregator<ExtNetworkRequest>(nullptr, "requests"); | ||||||
|  |     m_extQuotes = new ExtItemAggregator<ExtQuotes>(nullptr, "quotes"); | ||||||
|  |     m_extScripts = new ExtItemAggregator<ExtScript>(nullptr, "scripts"); | ||||||
|  |     m_extUpgrade = new ExtItemAggregator<ExtUpgrade>(nullptr, "upgrade"); | ||||||
|  |     m_extWeather = new ExtItemAggregator<ExtWeather>(nullptr, "weather"); | ||||||
|  | } | ||||||
|  |  | ||||||
|  |  | ||||||
|  | AWKeyOperations::~AWKeyOperations() | ||||||
|  | { | ||||||
|  |     qCDebug(LOG_AW) << __PRETTY_FUNCTION__; | ||||||
|  | } | ||||||
|  |  | ||||||
|  |  | ||||||
|  | QStringList AWKeyOperations::devices(const QString &_type) const | ||||||
|  | { | ||||||
|  |     qCDebug(LOG_AW) << "Looking for type" << _type; | ||||||
|  |  | ||||||
|  |     return m_devices[_type]; | ||||||
|  | } | ||||||
|  |  | ||||||
|  |  | ||||||
|  | QHash<QString, QStringList> AWKeyOperations::devices() const | ||||||
|  | { | ||||||
|  |     return m_devices; | ||||||
|  | } | ||||||
|  |  | ||||||
|  |  | ||||||
|  | void AWKeyOperations::updateCache() | ||||||
|  | { | ||||||
|  |     // update network and hdd list | ||||||
|  |     addKeyToCache("hdd"); | ||||||
|  |     addKeyToCache("net"); | ||||||
|  | } | ||||||
|  |  | ||||||
|  |  | ||||||
|  | QStringList AWKeyOperations::dictKeys() const | ||||||
|  | { | ||||||
|  |     QStringList allKeys; | ||||||
|  |     // weather | ||||||
|  |     for (auto &item : m_extWeather->activeItems()) { | ||||||
|  |         allKeys.append(item->tag("weatherId")); | ||||||
|  |         allKeys.append(item->tag("weather")); | ||||||
|  |         allKeys.append(item->tag("humidity")); | ||||||
|  |         allKeys.append(item->tag("pressure")); | ||||||
|  |         allKeys.append(item->tag("temperature")); | ||||||
|  |         allKeys.append(item->tag("timestamp")); | ||||||
|  |     } | ||||||
|  |     // cpuclock & cpu | ||||||
|  |     for (int i = 0; i < QThread::idealThreadCount(); i++) { | ||||||
|  |         allKeys.append(QString("cpucl%1").arg(i)); | ||||||
|  |         allKeys.append(QString("cpu%1").arg(i)); | ||||||
|  |     } | ||||||
|  |     // temperature | ||||||
|  |     for (int i = 0; i < m_devices["temp"].count(); i++) | ||||||
|  |         allKeys.append(QString("temp%1").arg(i)); | ||||||
|  |     // hdd | ||||||
|  |     for (int i = 0; i < m_devices["mount"].count(); i++) { | ||||||
|  |         allKeys.append(QString("hddmb%1").arg(i)); | ||||||
|  |         allKeys.append(QString("hddgb%1").arg(i)); | ||||||
|  |         allKeys.append(QString("hddfreemb%1").arg(i)); | ||||||
|  |         allKeys.append(QString("hddfreegb%1").arg(i)); | ||||||
|  |         allKeys.append(QString("hddtotmb%1").arg(i)); | ||||||
|  |         allKeys.append(QString("hddtotgb%1").arg(i)); | ||||||
|  |         allKeys.append(QString("hdd%1").arg(i)); | ||||||
|  |     } | ||||||
|  |     // hdd speed | ||||||
|  |     for (int i = 0; i < m_devices["disk"].count(); i++) { | ||||||
|  |         allKeys.append(QString("hddr%1").arg(i)); | ||||||
|  |         allKeys.append(QString("hddw%1").arg(i)); | ||||||
|  |     } | ||||||
|  |     // hdd temp | ||||||
|  |     for (int i = 0; i < m_devices["hdd"].count(); i++) | ||||||
|  |         allKeys.append(QString("hddtemp%1").arg(i)); | ||||||
|  |     // network | ||||||
|  |     for (int i = 0; i < m_devices["net"].count(); i++) { | ||||||
|  |         allKeys.append(QString("downunits%1").arg(i)); | ||||||
|  |         allKeys.append(QString("upunits%1").arg(i)); | ||||||
|  |         allKeys.append(QString("downtotkb%1").arg(i)); | ||||||
|  |         allKeys.append(QString("downtot%1").arg(i)); | ||||||
|  |         allKeys.append(QString("downkb%1").arg(i)); | ||||||
|  |         allKeys.append(QString("down%1").arg(i)); | ||||||
|  |         allKeys.append(QString("uptotkb%1").arg(i)); | ||||||
|  |         allKeys.append(QString("uptot%1").arg(i)); | ||||||
|  |         allKeys.append(QString("upkb%1").arg(i)); | ||||||
|  |         allKeys.append(QString("up%1").arg(i)); | ||||||
|  |     } | ||||||
|  |     // battery | ||||||
|  |     QStringList allBatteryDevices | ||||||
|  |         = QDir("/sys/class/power_supply") | ||||||
|  |               .entryList(QStringList({"BAT*"}), QDir::Dirs | QDir::NoDotAndDotDot, QDir::Name); | ||||||
|  |     for (int i = 0; i < allBatteryDevices.count(); i++) { | ||||||
|  |         allKeys.append(QString("bat%1").arg(i)); | ||||||
|  |         allKeys.append(QString("batleft%1").arg(i)); | ||||||
|  |         allKeys.append(QString("batnow%1").arg(i)); | ||||||
|  |         allKeys.append(QString("batrate%1").arg(i)); | ||||||
|  |         allKeys.append(QString("battotal%1").arg(i)); | ||||||
|  |     } | ||||||
|  |     // package manager | ||||||
|  |     for (auto &item : m_extUpgrade->activeItems()) | ||||||
|  |         allKeys.append(item->tag("pkgcount")); | ||||||
|  |     // quotes | ||||||
|  |     for (auto &item : m_extQuotes->activeItems()) { | ||||||
|  |         allKeys.append(item->tag("price")); | ||||||
|  |         allKeys.append(item->tag("pricechg")); | ||||||
|  |         allKeys.append(item->tag("percpricechg")); | ||||||
|  |         allKeys.append(item->tag("volume")); | ||||||
|  |         allKeys.append(item->tag("volumechg")); | ||||||
|  |         allKeys.append(item->tag("percvolumechg")); | ||||||
|  |     } | ||||||
|  |     // custom | ||||||
|  |     for (auto &item : m_extScripts->activeItems()) | ||||||
|  |         allKeys.append(item->tag("custom")); | ||||||
|  |     // network requests | ||||||
|  |     for (auto &item : m_extNetRequest->activeItems()) | ||||||
|  |         allKeys.append(item->tag("response")); | ||||||
|  |     // bars | ||||||
|  |     for (auto &item : m_graphicalItems->activeItems()) | ||||||
|  |         allKeys.append(item->tag("bar")); | ||||||
|  |     // user defined keys | ||||||
|  |     allKeys.append(m_customKeys->keys()); | ||||||
|  |     // static keys | ||||||
|  |     allKeys.append(QString(STATIC_KEYS).split(',')); | ||||||
|  |  | ||||||
|  |     // sort in valid order | ||||||
|  |     allKeys.sort(); | ||||||
|  |     std::reverse(allKeys.begin(), allKeys.end()); | ||||||
|  |  | ||||||
|  |     return allKeys; | ||||||
|  | } | ||||||
|  |  | ||||||
|  |  | ||||||
|  | // this method is required to provide GraphicalItem functions (e.g. paint()) to | ||||||
|  | // parent classes | ||||||
|  | GraphicalItem *AWKeyOperations::giByKey(const QString &_key) const | ||||||
|  | { | ||||||
|  |     qCDebug(LOG_AW) << "Looking for item" << _key; | ||||||
|  |  | ||||||
|  |     return m_graphicalItems->itemByTag(_key, "bar"); | ||||||
|  | } | ||||||
|  |  | ||||||
|  |  | ||||||
|  | QStringList AWKeyOperations::requiredUserKeys() const | ||||||
|  | { | ||||||
|  |     return m_customKeys->refinedSources(); | ||||||
|  | } | ||||||
|  |  | ||||||
|  |  | ||||||
|  | QStringList AWKeyOperations::userKeys() const | ||||||
|  | { | ||||||
|  |     return m_customKeys->keys(); | ||||||
|  | } | ||||||
|  |  | ||||||
|  |  | ||||||
|  | QString AWKeyOperations::userKeySource(const QString &_key) const | ||||||
|  | { | ||||||
|  |     return m_customKeys->source(_key); | ||||||
|  | } | ||||||
|  |  | ||||||
|  |  | ||||||
|  | QString AWKeyOperations::infoByKey(const QString &_key) const | ||||||
|  | { | ||||||
|  |     qCDebug(LOG_AW) << "Requested key" << _key; | ||||||
|  |  | ||||||
|  |     QString stripped = _key; | ||||||
|  |     stripped.remove(QRegExp("\\d+")); | ||||||
|  |     QString output; | ||||||
|  |  | ||||||
|  |     if (_key.startsWith("bar")) { | ||||||
|  |         AbstractExtItem *item = m_graphicalItems->itemByTag(_key, stripped); | ||||||
|  |         if (item) | ||||||
|  |             output = item->uniq(); | ||||||
|  |     } else if (_key.startsWith("custom")) { | ||||||
|  |         AbstractExtItem *item = m_extScripts->itemByTag(_key, stripped); | ||||||
|  |         if (item) | ||||||
|  |             output = item->uniq(); | ||||||
|  |     } else if (_key.contains(QRegExp("^hdd[rw]"))) { | ||||||
|  |         QString index = _key; | ||||||
|  |         index.remove(QRegExp("hdd[rw]")); | ||||||
|  |         output = m_devices["disk"][index.toInt()]; | ||||||
|  |     } else if (_key.contains(QRegExp("^hdd([0-9]|mb|gb|freemb|freegb|totmb|totgb)"))) { | ||||||
|  |         QString index = _key; | ||||||
|  |         index.remove(QRegExp("^hdd(|mb|gb|freemb|freegb|totmb|totgb)")); | ||||||
|  |         output = m_devices["mount"][index.toInt()]; | ||||||
|  |     } else if (_key.startsWith("hddtemp")) { | ||||||
|  |         QString index = _key; | ||||||
|  |         index.remove("hddtemp"); | ||||||
|  |         output = m_devices["hdd"][index.toInt()]; | ||||||
|  |     } else if (_key.contains(QRegExp("^(down|up)[0-9]"))) { | ||||||
|  |         QString index = _key; | ||||||
|  |         index.remove(QRegExp("^(down|up)")); | ||||||
|  |         output = m_devices["net"][index.toInt()]; | ||||||
|  |     } else if (_key.startsWith("pkgcount")) { | ||||||
|  |         AbstractExtItem *item = m_extUpgrade->itemByTag(_key, stripped); | ||||||
|  |         if (item) | ||||||
|  |             output = item->uniq(); | ||||||
|  |     } else if (_key.contains(QRegExp("(^|perc)(ask|bid|price)(chg|)"))) { | ||||||
|  |         AbstractExtItem *item = m_extQuotes->itemByTag(_key, stripped); | ||||||
|  |         if (item) | ||||||
|  |             output = item->uniq(); | ||||||
|  |     } else if (_key.contains(QRegExp("(weather|weatherId|humidity|pressure|temperature)"))) { | ||||||
|  |         AbstractExtItem *item = m_extWeather->itemByTag(_key, stripped); | ||||||
|  |         if (item) | ||||||
|  |             output = item->uniq(); | ||||||
|  |     } else if (_key.startsWith("temp")) { | ||||||
|  |         QString index = _key; | ||||||
|  |         index.remove("temp"); | ||||||
|  |         output = m_devices["temp"][index.toInt()]; | ||||||
|  |     } else if (_key.startsWith("response")) { | ||||||
|  |         AbstractExtItem *item = m_extNetRequest->itemByTag(_key, stripped); | ||||||
|  |         if (item) | ||||||
|  |             output = item->uniq(); | ||||||
|  |     } else { | ||||||
|  |         output = "(none)"; | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     return output; | ||||||
|  | } | ||||||
|  |  | ||||||
|  |  | ||||||
|  | QString AWKeyOperations::pattern() const | ||||||
|  | { | ||||||
|  |     return m_pattern; | ||||||
|  | } | ||||||
|  |  | ||||||
|  |  | ||||||
|  | void AWKeyOperations::setPattern(const QString &_currentPattern) | ||||||
|  | { | ||||||
|  |     qCDebug(LOG_AW) << "Set pattern" << _currentPattern; | ||||||
|  |  | ||||||
|  |     m_pattern = _currentPattern; | ||||||
|  | } | ||||||
|  |  | ||||||
|  |  | ||||||
|  | void AWKeyOperations::editItem(const QString &_type) | ||||||
|  | { | ||||||
|  |     qCDebug(LOG_AW) << "Item type" << _type; | ||||||
|  |  | ||||||
|  |     if (_type == "graphicalitem") { | ||||||
|  |         QStringList keys = dictKeys().filter(QRegExp("^(cpu(?!cl).*|gpu$|mem$|swap$|hdd[0-9].*|bat.*)")); | ||||||
|  |         keys.sort(); | ||||||
|  |         m_graphicalItems->setConfigArgs(keys); | ||||||
|  |         return m_graphicalItems->editItems(); | ||||||
|  |     } else if (_type == "extnetworkrequest") { | ||||||
|  |         return m_extNetRequest->editItems(); | ||||||
|  |     } else if (_type == "extquotes") { | ||||||
|  |         return m_extQuotes->editItems(); | ||||||
|  |     } else if (_type == "extscript") { | ||||||
|  |         return m_extScripts->editItems(); | ||||||
|  |     } else if (_type == "extupgrade") { | ||||||
|  |         return m_extUpgrade->editItems(); | ||||||
|  |     } else if (_type == "extweather") { | ||||||
|  |         return m_extWeather->editItems(); | ||||||
|  |     } | ||||||
|  | } | ||||||
|  |  | ||||||
|  |  | ||||||
|  | void AWKeyOperations::addDevice(const QString &_source) | ||||||
|  | { | ||||||
|  |     qCDebug(LOG_AW) << "Source" << _source; | ||||||
|  |  | ||||||
|  |     QRegExp diskRegexp = QRegExp("disk/(?:md|sd|hd)[a-z|0-9]_.*/Rate/(?:rblk)"); | ||||||
|  |     QRegExp mountRegexp = QRegExp("partitions/.*/filllevel"); | ||||||
|  |  | ||||||
|  |     if (_source.contains(diskRegexp)) { | ||||||
|  |         QString device = _source; | ||||||
|  |         device.remove("/Rate/rblk"); | ||||||
|  |         addKeyToCache("disk", device); | ||||||
|  |     } else if (_source.contains(mountRegexp)) { | ||||||
|  |         QString device = _source; | ||||||
|  |         device.remove("partitions").remove("/filllevel"); | ||||||
|  |         addKeyToCache("mount", device); | ||||||
|  |     } else if (_source.startsWith("lmsensors")) { | ||||||
|  |         addKeyToCache("temp", _source); | ||||||
|  |     } | ||||||
|  | } | ||||||
|  |  | ||||||
|  |  | ||||||
|  | void AWKeyOperations::addKeyToCache(const QString &_type, const QString &_key) | ||||||
|  | { | ||||||
|  |     qCDebug(LOG_AW) << "Key" << _key << "with type" << _type; | ||||||
|  |  | ||||||
|  |     if (AWKeyCache::addKeyToCache(_type, _key)) { | ||||||
|  |         m_devices = AWKeyCache::loadKeysFromCache(); | ||||||
|  |         reinitKeys(); | ||||||
|  |     } | ||||||
|  | } | ||||||
|  |  | ||||||
|  |  | ||||||
|  | void AWKeyOperations::reinitKeys() | ||||||
|  | { | ||||||
|  |     m_customKeys->initItems(); | ||||||
|  |     m_graphicalItems->initItems(); | ||||||
|  |     m_extNetRequest->initItems(); | ||||||
|  |     m_extQuotes->initItems(); | ||||||
|  |     m_extScripts->initItems(); | ||||||
|  |     m_extUpgrade->initItems(); | ||||||
|  |     m_extWeather->initItems(); | ||||||
|  |  | ||||||
|  |     // init | ||||||
|  |     QStringList allKeys = dictKeys(); | ||||||
|  |  | ||||||
|  |     // apply aw_* functions | ||||||
|  |     m_pattern = AWPatternFunctions::insertAllKeys(m_pattern, allKeys); | ||||||
|  |     m_pattern = AWPatternFunctions::insertKeyCount(m_pattern, allKeys); | ||||||
|  |     m_pattern = AWPatternFunctions::insertKeyNames(m_pattern, allKeys); | ||||||
|  |     m_pattern = AWPatternFunctions::insertKeys(m_pattern, allKeys); | ||||||
|  |     m_pattern = AWPatternFunctions::insertMacros(m_pattern); | ||||||
|  |     // wrap templates | ||||||
|  |     m_pattern = AWPatternFunctions::expandTemplates(m_pattern); | ||||||
|  |  | ||||||
|  |     emit(updateKeys(allKeys)); | ||||||
|  | } | ||||||
							
								
								
									
										83
									
								
								sources/awesome-widget/plugin/awkeyoperations.h
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										83
									
								
								sources/awesome-widget/plugin/awkeyoperations.h
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,83 @@ | |||||||
|  | /*************************************************************************** | ||||||
|  |  *   This file is part of awesome-widgets                                  * | ||||||
|  |  *                                                                         * | ||||||
|  |  *   awesome-widgets is free software: you can redistribute it and/or      * | ||||||
|  |  *   modify it under the terms of the GNU General Public License as        * | ||||||
|  |  *   published by the Free Software Foundation, either version 3 of the    * | ||||||
|  |  *   License, or (at your option) any later version.                       * | ||||||
|  |  *                                                                         * | ||||||
|  |  *   awesome-widgets is distributed in the hope that it will be useful,    * | ||||||
|  |  *   but WITHOUT ANY WARRANTY; without even the implied warranty of        * | ||||||
|  |  *   MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the         * | ||||||
|  |  *   GNU General Public License for more details.                          * | ||||||
|  |  *                                                                         * | ||||||
|  |  *   You should have received a copy of the GNU General Public License     * | ||||||
|  |  *   along with awesome-widgets. If not, see http://www.gnu.org/licenses/  * | ||||||
|  |  ***************************************************************************/ | ||||||
|  |  | ||||||
|  |  | ||||||
|  | #ifndef AWKEYOPERATIONS_H | ||||||
|  | #define AWKEYOPERATIONS_H | ||||||
|  |  | ||||||
|  | #include <QObject> | ||||||
|  |  | ||||||
|  | #include "extitemaggregator.h" | ||||||
|  |  | ||||||
|  |  | ||||||
|  | class AWCustomKeysHelper; | ||||||
|  | class ExtNetworkRequest; | ||||||
|  | class ExtQuotes; | ||||||
|  | class ExtScript; | ||||||
|  | class ExtUpgrade; | ||||||
|  | class ExtWeather; | ||||||
|  | class GraphicalItem; | ||||||
|  |  | ||||||
|  | class AWKeyOperations : public QObject | ||||||
|  | { | ||||||
|  |     Q_OBJECT | ||||||
|  |     Q_PROPERTY(QString pattern READ pattern WRITE setPattern) | ||||||
|  |  | ||||||
|  | public: | ||||||
|  |     explicit AWKeyOperations(QObject *_parent = nullptr); | ||||||
|  |     ~AWKeyOperations() override; | ||||||
|  |     [[nodiscard]] QStringList devices(const QString &_type) const; | ||||||
|  |     [[nodiscard]] QHash<QString, QStringList> devices() const; | ||||||
|  |     void updateCache(); | ||||||
|  |     // keys | ||||||
|  |     [[nodiscard]] QStringList dictKeys() const; | ||||||
|  |     [[nodiscard]] GraphicalItem *giByKey(const QString &_key) const; | ||||||
|  |     [[nodiscard]] QStringList requiredUserKeys() const; | ||||||
|  |     [[nodiscard]] QStringList userKeys() const; | ||||||
|  |     [[nodiscard]] QString userKeySource(const QString &_key) const; | ||||||
|  |     // values | ||||||
|  |     [[nodiscard]] QString infoByKey(const QString &_key) const; | ||||||
|  |     [[nodiscard]] QString pattern() const; | ||||||
|  |     void setPattern(const QString &_currentPattern); | ||||||
|  |     // configuration | ||||||
|  |     void editItem(const QString &_type); | ||||||
|  |  | ||||||
|  | signals: | ||||||
|  |     void updateKeys(const QStringList &_currentKeys); | ||||||
|  |  | ||||||
|  | public slots: | ||||||
|  |     void addDevice(const QString &_source); | ||||||
|  |  | ||||||
|  | private: | ||||||
|  |     // methods | ||||||
|  |     void addKeyToCache(const QString &_type, const QString &_key = ""); | ||||||
|  |     void reinitKeys(); | ||||||
|  |     // objects | ||||||
|  |     AWCustomKeysHelper *m_customKeys = nullptr; | ||||||
|  |     ExtItemAggregator<GraphicalItem> *m_graphicalItems = nullptr; | ||||||
|  |     ExtItemAggregator<ExtNetworkRequest> *m_extNetRequest = nullptr; | ||||||
|  |     ExtItemAggregator<ExtQuotes> *m_extQuotes = nullptr; | ||||||
|  |     ExtItemAggregator<ExtScript> *m_extScripts = nullptr; | ||||||
|  |     ExtItemAggregator<ExtUpgrade> *m_extUpgrade = nullptr; | ||||||
|  |     ExtItemAggregator<ExtWeather> *m_extWeather = nullptr; | ||||||
|  |     // variables | ||||||
|  |     QHash<QString, QStringList> m_devices; | ||||||
|  |     QString m_pattern; | ||||||
|  | }; | ||||||
|  |  | ||||||
|  |  | ||||||
|  | #endif /* AWKEYOPERATIONS_H */ | ||||||
| @ -17,47 +17,55 @@ | |||||||
|  |  | ||||||
| #include "awkeys.h" | #include "awkeys.h" | ||||||
|  |  | ||||||
| #include <QtConcurrent/QtConcurrent> | #include <QDBusConnection> | ||||||
| #include <QDir> | #include <QDBusError> | ||||||
| #include <QInputDialog> |  | ||||||
| #include <QJSEngine> |  | ||||||
| #include <QNetworkInterface> |  | ||||||
| #include <QRegExp> |  | ||||||
| #include <QSettings> |  | ||||||
| #include <QStandardPaths> |  | ||||||
| #include <QThread> | #include <QThread> | ||||||
|  | #include <QTimer> | ||||||
|  | #include <QtConcurrent/QtConcurrent> | ||||||
|  |  | ||||||
| #include "awdataaggregator.h" | #include "awdataaggregator.h" | ||||||
| #include "awdataengineaggregator.h" | #include "awdataengineaggregator.h" | ||||||
|  | #include "awdbusadaptor.h" | ||||||
| #include "awdebug.h" | #include "awdebug.h" | ||||||
|  | #include "awkeycache.h" | ||||||
|  | #include "awkeyoperations.h" | ||||||
| #include "awkeysaggregator.h" | #include "awkeysaggregator.h" | ||||||
| #include "extquotes.h" | #include "awpatternfunctions.h" | ||||||
| #include "extscript.h" |  | ||||||
| #include "extupgrade.h" |  | ||||||
| #include "extweather.h" |  | ||||||
| #include "graphicalitem.h" | #include "graphicalitem.h" | ||||||
| #include "version.h" |  | ||||||
|  |  | ||||||
|  |  | ||||||
| AWKeys::AWKeys(QObject *parent) | AWKeys::AWKeys(QObject *_parent) | ||||||
|     : QObject(parent) |     : QObject(_parent) | ||||||
| { | { | ||||||
|     qSetMessagePattern(LOG_FORMAT); |     qSetMessagePattern(AWDebug::LOG_FORMAT); | ||||||
|     qCDebug(LOG_AW) << __PRETTY_FUNCTION__; |     qCDebug(LOG_AW) << __PRETTY_FUNCTION__; | ||||||
|     foreach (const QString metadata, getBuildData()) |     for (auto &metadata : AWDebug::getBuildData()) | ||||||
|         qCDebug(LOG_AW) << metadata; |         qCDebug(LOG_AW) << metadata; | ||||||
|  |  | ||||||
| #ifdef BUILD_FUTURE |  | ||||||
|     // thread pool |     // thread pool | ||||||
|     m_threadPool = new QThreadPool(this); |     m_threadPool = new QThreadPool(this); | ||||||
| #endif /* BUILD_FUTURE */ |  | ||||||
|  |  | ||||||
|     aggregator = new AWKeysAggregator(this); |     m_aggregator = new AWKeysAggregator(this); | ||||||
|     dataAggregator = new AWDataAggregator(this); |     m_dataAggregator = new AWDataAggregator(this); | ||||||
|  |     m_dataEngineAggregator = new AWDataEngineAggregator(this); | ||||||
|  |     m_keyOperator = new AWKeyOperations(this); | ||||||
|  |  | ||||||
|  |     m_timer = new QTimer(this); | ||||||
|  |     m_timer->setSingleShot(false); | ||||||
|  |  | ||||||
|  |     createDBusInterface(); | ||||||
|  |  | ||||||
|  |     // update key data if required | ||||||
|  |     connect(m_keyOperator, SIGNAL(updateKeys(const QStringList &)), this, SLOT(reinitKeys(const QStringList &))); | ||||||
|  |     connect(m_timer, SIGNAL(timeout()), this, SLOT(updateTextData())); | ||||||
|     // transfer signal from AWDataAggregator object to QML ui |     // transfer signal from AWDataAggregator object to QML ui | ||||||
|     connect(dataAggregator, SIGNAL(toolTipPainted(const QString)), this, |     connect(m_dataAggregator, SIGNAL(toolTipPainted(const QString &)), this, | ||||||
|             SIGNAL(needToolTipToBeUpdated(const QString))); |             SIGNAL(needToolTipToBeUpdated(const QString &))); | ||||||
|     connect(this, SIGNAL(needToBeUpdated()), this, SLOT(updateTextData())); |     connect(this, SIGNAL(dropSourceFromDataengine(const QString &)), m_dataEngineAggregator, | ||||||
|  |             SLOT(dropSource(const QString &))); | ||||||
|  |     // transfer signal from dataengine to update source list | ||||||
|  |     connect(m_dataEngineAggregator, SIGNAL(deviceAdded(const QString &)), m_keyOperator, | ||||||
|  |             SLOT(addDevice(const QString &))); | ||||||
| } | } | ||||||
|  |  | ||||||
|  |  | ||||||
| @ -65,572 +73,178 @@ AWKeys::~AWKeys() | |||||||
| { | { | ||||||
|     qCDebug(LOG_AW) << __PRETTY_FUNCTION__; |     qCDebug(LOG_AW) << __PRETTY_FUNCTION__; | ||||||
|  |  | ||||||
|     // extensions |     m_timer->stop(); | ||||||
|     if (graphicalItems != nullptr) |     // delete dbus session | ||||||
|         delete graphicalItems; |     auto id = reinterpret_cast<qlonglong>(this); | ||||||
|     if (extQuotes != nullptr) |     QDBusConnection::sessionBus().unregisterObject(QString("/%1").arg(id)); | ||||||
|         delete extQuotes; |  | ||||||
|     if (extScripts != nullptr) |  | ||||||
|         delete extScripts; |  | ||||||
|     if (extUpgrade != nullptr) |  | ||||||
|         delete extUpgrade; |  | ||||||
|     if (extWeather != nullptr) |  | ||||||
|         delete extWeather; |  | ||||||
|  |  | ||||||
|     // core |  | ||||||
|     if (dataEngineAggregator != nullptr) |  | ||||||
|         delete dataEngineAggregator; |  | ||||||
|     if (m_threadPool != nullptr) |  | ||||||
|         delete m_threadPool; |  | ||||||
|     delete aggregator; |  | ||||||
|     delete dataAggregator; |  | ||||||
| } | } | ||||||
|  |  | ||||||
|  |  | ||||||
| void AWKeys::initDataAggregator(const QVariantMap tooltipParams) | void AWKeys::initDataAggregator(const QVariantMap &_tooltipParams) | ||||||
| { | { | ||||||
|     qCDebug(LOG_AW) << "Tooltip parameters" << tooltipParams; |     qCDebug(LOG_AW) << "Tooltip parameters" << _tooltipParams; | ||||||
|  |  | ||||||
|     dataAggregator->setParameters(tooltipParams); |     // store parameters to generate m_requiredKeys | ||||||
|  |     m_tooltipParams = _tooltipParams; | ||||||
|  |     m_dataAggregator->setParameters(m_tooltipParams); | ||||||
| } | } | ||||||
|  |  | ||||||
|  |  | ||||||
| void AWKeys::initKeys(const QString currentPattern, const int interval, | void AWKeys::initKeys(const QString &_currentPattern, const int _interval, const int _limit, const bool _optimize) | ||||||
|                       const int limit) |  | ||||||
| { | { | ||||||
|     qCDebug(LOG_AW) << "Pattern" << currentPattern; |     qCDebug(LOG_AW) << "Pattern" << _currentPattern << "with interval" << _interval << "and queue limit" << _limit | ||||||
|     qCDebug(LOG_AW) << "Interval" << interval; |                     << "with optimization" << _optimize; | ||||||
|     qCDebug(LOG_AW) << "Queue limit" << limit; |  | ||||||
|  |  | ||||||
|     // init |     // init | ||||||
|     m_pattern = currentPattern; |     m_optimize = _optimize; | ||||||
|     if (dataEngineAggregator == nullptr) { |     m_threadPool->setMaxThreadCount(_limit == 0 ? QThread::idealThreadCount() : _limit); | ||||||
|         dataEngineAggregator = new AWDataEngineAggregator(this, interval); |     // child objects | ||||||
|         connect(this, SIGNAL(dropSourceFromDataengine(QString)), |     m_aggregator->initFormatters(); | ||||||
|                 dataEngineAggregator, SLOT(dropSource(QString))); |     m_keyOperator->setPattern(_currentPattern); | ||||||
|     } else |     m_keyOperator->updateCache(); | ||||||
|         dataEngineAggregator->setInterval(interval); |     m_dataEngineAggregator->reconnectSources(_interval); | ||||||
| #ifdef BUILD_FUTURE |  | ||||||
|     m_threadPool->setMaxThreadCount(limit == 0 ? QThread::idealThreadCount() |  | ||||||
|                                                : limit); |  | ||||||
| #endif /* BUILD_FUTURE */ |  | ||||||
|     updateCache(); |  | ||||||
|  |  | ||||||
|     return dataEngineAggregator->reconnectSources(); |     // timer | ||||||
|  |     m_timer->setInterval(_interval); | ||||||
|  |     m_timer->start(); | ||||||
| } | } | ||||||
|  |  | ||||||
|  |  | ||||||
| void AWKeys::setAggregatorProperty(const QString key, const QVariant value) | void AWKeys::setAggregatorProperty(const QString &_key, const QVariant &_value) | ||||||
| { | { | ||||||
|     qCDebug(LOG_AW) << "Key" << key; |     qCDebug(LOG_AW) << "Key" << _key << "with value" << _value; | ||||||
|     qCDebug(LOG_AW) << "Value" << value; |  | ||||||
|  |  | ||||||
|     aggregator->setProperty(key.toUtf8().constData(), value); |     m_aggregator->setProperty(_key.toUtf8().constData(), _value); | ||||||
| } | } | ||||||
|  |  | ||||||
|  |  | ||||||
| void AWKeys::setWrapNewLines(const bool wrap) | void AWKeys::setWrapNewLines(const bool _wrap) | ||||||
| { | { | ||||||
|     qCDebug(LOG_AW) << "Is wrapping enabled" << wrap; |     qCDebug(LOG_AW) << "Is wrapping enabled" << _wrap; | ||||||
|  |  | ||||||
|     m_wrapNewLines = wrap; |     m_wrapNewLines = _wrap; | ||||||
| } | } | ||||||
|  |  | ||||||
|  |  | ||||||
| void AWKeys::updateCache() | void AWKeys::updateCache() | ||||||
| { | { | ||||||
|     // update network and hdd list |     return m_keyOperator->updateCache(); | ||||||
|     addKeyToCache(QString("hdd")); |  | ||||||
|     addKeyToCache(QString("net")); |  | ||||||
| } | } | ||||||
|  |  | ||||||
|  |  | ||||||
| QStringList AWKeys::dictKeys(const bool sorted, const QString regexp) const | QStringList AWKeys::dictKeys(const bool _sorted, const QString &_regexp) const | ||||||
| { | { | ||||||
|     qCDebug(LOG_AW) << "Should be sorted" << sorted; |     qCDebug(LOG_AW) << "Should be sorted" << _sorted << "and filter applied" << _regexp; | ||||||
|     qCDebug(LOG_AW) << "Filter" << regexp; |  | ||||||
|  |  | ||||||
|     QStringList allKeys; |     // check if functions asked | ||||||
|     // weather |     if (_regexp == "functions") | ||||||
|     for (int i = extWeather->activeItems().count() - 1; i >= 0; i--) { |         return QString(STATIC_FUNCTIONS).split(','); | ||||||
|         allKeys.append( |     // check if user defined keys asked | ||||||
|             extWeather->activeItems().at(i)->tag(QString("weatherId"))); |     if (_regexp == "userdefined") | ||||||
|         allKeys.append( |         return m_keyOperator->userKeys(); | ||||||
|             extWeather->activeItems().at(i)->tag(QString("weather"))); |  | ||||||
|         allKeys.append( |  | ||||||
|             extWeather->activeItems().at(i)->tag(QString("humidity"))); |  | ||||||
|         allKeys.append( |  | ||||||
|             extWeather->activeItems().at(i)->tag(QString("pressure"))); |  | ||||||
|         allKeys.append( |  | ||||||
|             extWeather->activeItems().at(i)->tag(QString("temperature"))); |  | ||||||
|         allKeys.append( |  | ||||||
|             extWeather->activeItems().at(i)->tag(QString("timestamp"))); |  | ||||||
|     } |  | ||||||
|     // time |  | ||||||
|     allKeys.append(QString("time")); |  | ||||||
|     allKeys.append(QString("isotime")); |  | ||||||
|     allKeys.append(QString("shorttime")); |  | ||||||
|     allKeys.append(QString("longtime")); |  | ||||||
|     allKeys.append(QString("ctime")); |  | ||||||
|     // uptime |  | ||||||
|     allKeys.append(QString("uptime")); |  | ||||||
|     allKeys.append(QString("cuptime")); |  | ||||||
|     // cpuclock & cpu |  | ||||||
|     for (int i = QThread::idealThreadCount() - 1; i >= 0; i--) { |  | ||||||
|         allKeys.append(QString("cpucl%1").arg(i)); |  | ||||||
|         allKeys.append(QString("cpu%1").arg(i)); |  | ||||||
|     } |  | ||||||
|     allKeys.append(QString("cpucl")); |  | ||||||
|     allKeys.append(QString("cpu")); |  | ||||||
|     // temperature |  | ||||||
|     for (int i = m_devices[QString("temp")].count() - 1; i >= 0; i--) |  | ||||||
|         allKeys.append(QString("temp%1").arg(i)); |  | ||||||
|     // gputemp |  | ||||||
|     allKeys.append(QString("gputemp")); |  | ||||||
|     // gpu |  | ||||||
|     allKeys.append(QString("gpu")); |  | ||||||
|     // memory |  | ||||||
|     allKeys.append(QString("memmb")); |  | ||||||
|     allKeys.append(QString("memgb")); |  | ||||||
|     allKeys.append(QString("memfreemb")); |  | ||||||
|     allKeys.append(QString("memfreegb")); |  | ||||||
|     allKeys.append(QString("memtotmb")); |  | ||||||
|     allKeys.append(QString("memtotgb")); |  | ||||||
|     allKeys.append(QString("memusedmb")); |  | ||||||
|     allKeys.append(QString("memusedgb")); |  | ||||||
|     allKeys.append(QString("mem")); |  | ||||||
|     // swap |  | ||||||
|     allKeys.append(QString("swapmb")); |  | ||||||
|     allKeys.append(QString("swapgb")); |  | ||||||
|     allKeys.append(QString("swapfreemb")); |  | ||||||
|     allKeys.append(QString("swapfreegb")); |  | ||||||
|     allKeys.append(QString("swaptotmb")); |  | ||||||
|     allKeys.append(QString("swaptotgb")); |  | ||||||
|     allKeys.append(QString("swap")); |  | ||||||
|     // hdd |  | ||||||
|     for (int i = m_devices[QString("mount")].count() - 1; i >= 0; i--) { |  | ||||||
|         allKeys.append(QString("hddmb%1").arg(i)); |  | ||||||
|         allKeys.append(QString("hddgb%1").arg(i)); |  | ||||||
|         allKeys.append(QString("hddfreemb%1").arg(i)); |  | ||||||
|         allKeys.append(QString("hddfreegb%1").arg(i)); |  | ||||||
|         allKeys.append(QString("hddtotmb%1").arg(i)); |  | ||||||
|         allKeys.append(QString("hddtotgb%1").arg(i)); |  | ||||||
|         allKeys.append(QString("hdd%1").arg(i)); |  | ||||||
|     } |  | ||||||
|     // hdd speed |  | ||||||
|     for (int i = m_devices[QString("disk")].count() - 1; i >= 0; i--) { |  | ||||||
|         allKeys.append(QString("hddr%1").arg(i)); |  | ||||||
|         allKeys.append(QString("hddw%1").arg(i)); |  | ||||||
|     } |  | ||||||
|     // hdd temp |  | ||||||
|     for (int i = m_devices[QString("hdd")].count() - 1; i >= 0; i--) |  | ||||||
|         allKeys.append(QString("hddtemp%1").arg(i)); |  | ||||||
|     // network |  | ||||||
|     for (int i = m_devices[QString("net")].count() - 1; i >= 0; i--) { |  | ||||||
|         allKeys.append(QString("downunits%1").arg(i)); |  | ||||||
|         allKeys.append(QString("upunits%1").arg(i)); |  | ||||||
|         allKeys.append(QString("downkb%1").arg(i)); |  | ||||||
|         allKeys.append(QString("down%1").arg(i)); |  | ||||||
|         allKeys.append(QString("upkb%1").arg(i)); |  | ||||||
|         allKeys.append(QString("up%1").arg(i)); |  | ||||||
|     } |  | ||||||
|     allKeys.append(QString("downunits")); |  | ||||||
|     allKeys.append(QString("upunits")); |  | ||||||
|     allKeys.append(QString("downkb")); |  | ||||||
|     allKeys.append(QString("down")); |  | ||||||
|     allKeys.append(QString("upkb")); |  | ||||||
|     allKeys.append(QString("up")); |  | ||||||
|     allKeys.append(QString("netdev")); |  | ||||||
|     // battery |  | ||||||
|     allKeys.append(QString("ac")); |  | ||||||
|     QStringList allBatteryDevices |  | ||||||
|         = QDir(QString("/sys/class/power_supply")) |  | ||||||
|               .entryList(QStringList() << QString("BAT*"), |  | ||||||
|                          QDir::Dirs | QDir::NoDotAndDotDot, QDir::Name); |  | ||||||
|     for (int i = allBatteryDevices.count() - 1; i >= 0; i--) |  | ||||||
|         allKeys.append(QString("bat%1").arg(i)); |  | ||||||
|     allKeys.append(QString("bat")); |  | ||||||
|     // player |  | ||||||
|     allKeys.append(QString("album")); |  | ||||||
|     allKeys.append(QString("artist")); |  | ||||||
|     allKeys.append(QString("duration")); |  | ||||||
|     allKeys.append(QString("progress")); |  | ||||||
|     allKeys.append(QString("title")); |  | ||||||
|     allKeys.append(QString("dalbum")); |  | ||||||
|     allKeys.append(QString("dartist")); |  | ||||||
|     allKeys.append(QString("dtitle")); |  | ||||||
|     allKeys.append(QString("salbum")); |  | ||||||
|     allKeys.append(QString("sartist")); |  | ||||||
|     allKeys.append(QString("stitle")); |  | ||||||
|     // ps |  | ||||||
|     allKeys.append(QString("pscount")); |  | ||||||
|     allKeys.append(QString("pstotal")); |  | ||||||
|     allKeys.append(QString("ps")); |  | ||||||
|     // package manager |  | ||||||
|     for (int i = extUpgrade->activeItems().count() - 1; i >= 0; i--) |  | ||||||
|         allKeys.append( |  | ||||||
|             extUpgrade->activeItems().at(i)->tag(QString("pkgcount"))); |  | ||||||
|     // quotes |  | ||||||
|     for (int i = extQuotes->activeItems().count() - 1; i >= 0; i--) { |  | ||||||
|         allKeys.append(extQuotes->activeItems().at(i)->tag(QString("ask"))); |  | ||||||
|         allKeys.append(extQuotes->activeItems().at(i)->tag(QString("askchg"))); |  | ||||||
|         allKeys.append( |  | ||||||
|             extQuotes->activeItems().at(i)->tag(QString("percaskchg"))); |  | ||||||
|         allKeys.append(extQuotes->activeItems().at(i)->tag(QString("bid"))); |  | ||||||
|         allKeys.append(extQuotes->activeItems().at(i)->tag(QString("bidchg"))); |  | ||||||
|         allKeys.append( |  | ||||||
|             extQuotes->activeItems().at(i)->tag(QString("percbidchg"))); |  | ||||||
|         allKeys.append(extQuotes->activeItems().at(i)->tag(QString("price"))); |  | ||||||
|         allKeys.append( |  | ||||||
|             extQuotes->activeItems().at(i)->tag(QString("pricechg"))); |  | ||||||
|         allKeys.append( |  | ||||||
|             extQuotes->activeItems().at(i)->tag(QString("percpricechg"))); |  | ||||||
|     } |  | ||||||
|     // custom |  | ||||||
|     for (int i = extScripts->activeItems().count() - 1; i >= 0; i--) |  | ||||||
|         allKeys.append(extScripts->activeItems().at(i)->tag(QString("custom"))); |  | ||||||
|     // desktop |  | ||||||
|     allKeys.append(QString("desktop")); |  | ||||||
|     allKeys.append(QString("ndesktop")); |  | ||||||
|     allKeys.append(QString("tdesktops")); |  | ||||||
|     // load average |  | ||||||
|     allKeys.append(QString("la15")); |  | ||||||
|     allKeys.append(QString("la5")); |  | ||||||
|     allKeys.append(QString("la1")); |  | ||||||
|     // bars |  | ||||||
|     QStringList graphicalItemsKeys; |  | ||||||
|     foreach (GraphicalItem *item, graphicalItems->items()) |  | ||||||
|         graphicalItemsKeys.append(item->tag()); |  | ||||||
|     graphicalItemsKeys.sort(); |  | ||||||
|     for (int i = graphicalItemsKeys.count() - 1; i >= 0; i--) |  | ||||||
|         allKeys.append(graphicalItemsKeys.at(i)); |  | ||||||
|  |  | ||||||
|  |     QStringList allKeys = m_keyOperator->dictKeys(); | ||||||
|     // sort if required |     // sort if required | ||||||
|     if (sorted) |     if (_sorted) | ||||||
|         allKeys.sort(); |         allKeys.sort(); | ||||||
|  |  | ||||||
|     return allKeys.filter(QRegExp(regexp)); |     return allKeys.filter(QRegExp(_regexp)); | ||||||
| } | } | ||||||
|  |  | ||||||
|  |  | ||||||
| QStringList AWKeys::getHddDevices() const | QVariantList AWKeys::getHddDevices() const | ||||||
| { | { | ||||||
|     QStringList devices = m_devices[QString("hdd")]; |     QStringList hddDevices = m_keyOperator->devices("hdd"); | ||||||
|     // required by selector in the UI |     // required by selector in the UI | ||||||
|     devices.insert(0, QString("disable")); |     hddDevices.insert(0, "disable"); | ||||||
|     devices.insert(0, QString("auto")); |     hddDevices.insert(0, "auto"); | ||||||
|  |  | ||||||
|  |     // build model | ||||||
|  |     QVariantList devices; | ||||||
|  |     for (auto &device : hddDevices) { | ||||||
|  |         QVariantMap model; | ||||||
|  |         model["label"] = device; | ||||||
|  |         model["name"] = device; | ||||||
|  |         devices.append(model); | ||||||
|  |     } | ||||||
|  |  | ||||||
|     return devices; |     return devices; | ||||||
| } | } | ||||||
|  |  | ||||||
|  |  | ||||||
| QString AWKeys::infoByKey(QString key) const | QString AWKeys::infoByKey(const QString &_key) const | ||||||
| { | { | ||||||
|     qCDebug(LOG_AW) << "Requested key" << key; |     qCDebug(LOG_AW) << "Requested info for key" << _key; | ||||||
|  |  | ||||||
|     key.remove(QRegExp(QString("^bar[0-9]{1,}"))); |     return m_keyOperator->infoByKey(_key); | ||||||
|     if (key.startsWith(QString("custom"))) |  | ||||||
|         return extScripts->itemByTagNumber( |  | ||||||
|                              key.remove(QString("custom")).toInt()) |  | ||||||
|             ->uniq(); |  | ||||||
|     else if (key.contains(QRegExp(QString("^hdd[rw]")))) |  | ||||||
|         return QString("%1").arg(m_devices[QString( |  | ||||||
|             "disk")][key.remove(QRegExp(QString("hdd[rw]"))).toInt()]); |  | ||||||
|     else if (key.contains(QRegExp( |  | ||||||
|                  QString("^hdd([0-9]|mb|gb|freemb|freegb|totmb|totgb)")))) |  | ||||||
|         return QString("%1").arg(m_devices[QString( |  | ||||||
|             "mount")][key |  | ||||||
|                           .remove(QRegExp(QString( |  | ||||||
|                               "^hdd([0-9]|mb|gb|freemb|freegb|totmb|totgb)"))) |  | ||||||
|                           .toInt()]); |  | ||||||
|     else if (key.startsWith(QString("hddtemp"))) |  | ||||||
|         return QString("%1").arg( |  | ||||||
|             m_devices[QString("hdd")][key.remove(QString("hddtemp")).toInt()]); |  | ||||||
|     else if (key.contains(QRegExp(QString("^(down|up)[0-9]")))) |  | ||||||
|         return QString("%1").arg(m_devices[QString( |  | ||||||
|             "net")][key.remove(QRegExp(QString("^(down|up)"))).toInt()]); |  | ||||||
|     else if (key.startsWith(QString("pkgcount"))) |  | ||||||
|         return extUpgrade->itemByTagNumber( |  | ||||||
|                              key.remove(QString("pkgcount")).toInt()) |  | ||||||
|             ->uniq(); |  | ||||||
|     else if (key.contains(QRegExp(QString("(^|perc)(ask|bid|price)(chg|)")))) |  | ||||||
|         return extQuotes->itemByTagNumber( |  | ||||||
|                             key.remove(QRegExp(QString( |  | ||||||
|                                            "(^|perc)(ask|bid|price)(chg|)"))) |  | ||||||
|                                 .toInt()) |  | ||||||
|             ->uniq(); |  | ||||||
|     else if (key.contains(QRegExp( |  | ||||||
|                  QString("(weather|weatherId|humidity|pressure|temperature)")))) |  | ||||||
|         return extWeather |  | ||||||
|             ->itemByTagNumber( |  | ||||||
|                 key |  | ||||||
|                     .remove(QRegExp(QString( |  | ||||||
|                         "(weather|weatherId|humidity|pressure|temperature)"))) |  | ||||||
|                     .toInt()) |  | ||||||
|             ->uniq(); |  | ||||||
|     else if (key.startsWith(QString("temp"))) |  | ||||||
|         return QString("%1").arg( |  | ||||||
|             m_devices[QString("temp")][key.remove(QString("temp")).toInt()]); |  | ||||||
|  |  | ||||||
|     return QString("(none)"); |  | ||||||
| } | } | ||||||
|  |  | ||||||
|  |  | ||||||
| // HACK this method requires to define tag value from bar from UI interface | // HACK this method requires to define tag value from bar from UI interface | ||||||
| QString AWKeys::valueByKey(QString key) const | QString AWKeys::valueByKey(const QString &_key) const | ||||||
| { | { | ||||||
|     qCDebug(LOG_AW) << "Requested key" << key; |     qCDebug(LOG_AW) << "Requested value for key" << _key; | ||||||
|  |  | ||||||
|     return values.value(key.remove(QRegExp(QString("^bar[0-9]{1,}"))), |     QString trueKey = _key.startsWith("bar") ? m_keyOperator->infoByKey(_key) : _key; | ||||||
|                         QString("")); |  | ||||||
|  |     return m_aggregator->formatter(m_values[trueKey], trueKey, true); | ||||||
| } | } | ||||||
|  |  | ||||||
|  |  | ||||||
| void AWKeys::editItem(const QString type) | void AWKeys::editItem(const QString &_type) | ||||||
| { | { | ||||||
|     qCDebug(LOG_AW) << "Item type" << type; |     qCDebug(LOG_AW) << "Item type" << _type; | ||||||
|  |  | ||||||
|     if (type == QString("graphicalitem")) { |     return m_keyOperator->editItem(_type); | ||||||
|         graphicalItems->setConfigArgs(dictKeys( |  | ||||||
|             true, QString("^(cpu(?!cl).*|gpu$|mem$|swap$|hdd[0-9].*|bat.*)"))); |  | ||||||
|         return graphicalItems->editItems(); |  | ||||||
|     } else if (type == QString("extquotes")) { |  | ||||||
|         return extQuotes->editItems(); |  | ||||||
|     } else if (type == QString("extscript")) { |  | ||||||
|         return extScripts->editItems(); |  | ||||||
|     } else if (type == QString("extupgrade")) { |  | ||||||
|         return extUpgrade->editItems(); |  | ||||||
|     } else if (type == QString("extweather")) { |  | ||||||
|         return extWeather->editItems(); |  | ||||||
|     } |  | ||||||
| } | } | ||||||
|  |  | ||||||
|  |  | ||||||
| void AWKeys::addDevice(const QString source) | void AWKeys::dataUpdated(const QString &_sourceName, const Plasma::DataEngine::Data &_data) | ||||||
| { | { | ||||||
|     qCDebug(LOG_AW) << "Source" << source; |  | ||||||
|  |  | ||||||
|     QRegExp diskRegexp |  | ||||||
|         = QRegExp(QString("disk/(?:md|sd|hd)[a-z|0-9]_.*/Rate/(?:rblk)")); |  | ||||||
|     QRegExp mountRegexp = QRegExp(QString("partitions/.*/filllevel")); |  | ||||||
|  |  | ||||||
|     if (source.contains(diskRegexp)) { |  | ||||||
|         QString device = source; |  | ||||||
|         device.remove(QString("/Rate/rblk")); |  | ||||||
|         addKeyToCache(QString("disk"), device); |  | ||||||
|     } else if (source.contains(mountRegexp)) { |  | ||||||
|         QString device = source; |  | ||||||
|         device.remove(QString("partitions")).remove(QString("/filllevel")); |  | ||||||
|         addKeyToCache(QString("mount"), device); |  | ||||||
|     } else if (source.startsWith(QString("lmsensors"))) { |  | ||||||
|         addKeyToCache(QString("temp"), source); |  | ||||||
|     } |  | ||||||
| } |  | ||||||
|  |  | ||||||
|  |  | ||||||
| void AWKeys::dataUpdated(const QString &sourceName, |  | ||||||
|                          const Plasma::DataEngine::Data &data) |  | ||||||
| { |  | ||||||
|     // do not log these parameters |  | ||||||
|     if (sourceName == QString("update")) { |  | ||||||
|         qCInfo(LOG_AW) << "Update data"; |  | ||||||
|         return emit(needToBeUpdated()); |  | ||||||
|     } |  | ||||||
|  |  | ||||||
| #ifdef BUILD_FUTURE |  | ||||||
|     // run concurrent data update |     // run concurrent data update | ||||||
|     QtConcurrent::run(m_threadPool, this, &AWKeys::setDataBySource, sourceName, |     QtConcurrent::run(m_threadPool, this, &AWKeys::setDataBySource, _sourceName, _data); | ||||||
|                       data); |  | ||||||
| #else  /* BUILD_FUTURE */ |  | ||||||
|     return setDataBySource(sourceName, data); |  | ||||||
| #endif /* BUILD_FUTURE */ |  | ||||||
| } | } | ||||||
|  |  | ||||||
|  |  | ||||||
| void AWKeys::loadKeysFromCache() | void AWKeys::reinitKeys(const QStringList &_currentKeys) | ||||||
| { | { | ||||||
|     QString fileName = QString("%1/awesomewidgets.ndx") |     qCDebug(LOG_AW) << "Update found keys by using list" << _currentKeys; | ||||||
|                            .arg(QStandardPaths::writableLocation( |  | ||||||
|                                QStandardPaths::GenericCacheLocation)); |  | ||||||
|     qCInfo(LOG_AW) << "Cache file" << fileName; |  | ||||||
|     QSettings cache(fileName, QSettings::IniFormat); |  | ||||||
|  |  | ||||||
|     foreach (QString group, cache.childGroups()) { |  | ||||||
|         cache.beginGroup(group); |  | ||||||
|         m_devices.remove(group); |  | ||||||
|         foreach (QString key, cache.allKeys()) |  | ||||||
|             m_devices[group].append(cache.value(key).toString()); |  | ||||||
|         cache.endGroup(); |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     return reinitKeys(); |  | ||||||
| } |  | ||||||
|  |  | ||||||
|  |  | ||||||
| void AWKeys::reinitKeys() |  | ||||||
| { |  | ||||||
|     // renew extensions |  | ||||||
|     // delete them if any |  | ||||||
|     if (graphicalItems != nullptr) |  | ||||||
|         delete graphicalItems; |  | ||||||
|     if (extQuotes != nullptr) |  | ||||||
|         delete extQuotes; |  | ||||||
|     if (extScripts != nullptr) |  | ||||||
|         delete extScripts; |  | ||||||
|     if (extUpgrade != nullptr) |  | ||||||
|         delete extUpgrade; |  | ||||||
|     if (extWeather != nullptr) |  | ||||||
|         delete extWeather; |  | ||||||
|     // create |  | ||||||
|     graphicalItems |  | ||||||
|         = new ExtItemAggregator<GraphicalItem>(nullptr, QString("desktops")); |  | ||||||
|     extQuotes = new ExtItemAggregator<ExtQuotes>(nullptr, QString("quotes")); |  | ||||||
|     extScripts = new ExtItemAggregator<ExtScript>(nullptr, QString("scripts")); |  | ||||||
|     extUpgrade = new ExtItemAggregator<ExtUpgrade>(nullptr, QString("upgrade")); |  | ||||||
|     extWeather = new ExtItemAggregator<ExtWeather>(nullptr, QString("weather")); |  | ||||||
|  |  | ||||||
|     // init |  | ||||||
|     QStringList allKeys = dictKeys(); |  | ||||||
|  |  | ||||||
| #ifdef BUILD_TESTING |  | ||||||
|     // not documented feature - place all available tags |  | ||||||
|     m_pattern = m_pattern.replace(QString("$ALL"), [allKeys]() { |  | ||||||
|         QStringList strings; |  | ||||||
|         foreach (QString tag, allKeys) |  | ||||||
|             strings.append(QString("%1: $%1").arg(tag)); |  | ||||||
|         return strings.join(QString(" | ")); |  | ||||||
|     }()); |  | ||||||
| #endif /* BUILD_TESTING */ |  | ||||||
|  |  | ||||||
|     // append lists |     // append lists | ||||||
|     // bars |     m_foundBars = AWPatternFunctions::findKeys(m_keyOperator->pattern(), _currentKeys, true); | ||||||
|     m_foundBars = [allKeys](QString pattern) { |     m_foundKeys = AWPatternFunctions::findKeys(m_keyOperator->pattern(), _currentKeys, false); | ||||||
|         QStringList selectedKeys; |     m_foundLambdas = AWPatternFunctions::findLambdas(m_keyOperator->pattern()); | ||||||
|         foreach (QString key, allKeys) |     // generate list of required keys for bars | ||||||
|             if ((key.startsWith(QString("bar"))) |     QStringList barKeys; | ||||||
|                 && (pattern.contains(QString("$%1").arg(key)))) { |     for (auto &bar : m_foundBars) { | ||||||
|                 qCInfo(LOG_AW) << "Found bar" << key; |         GraphicalItem *item = m_keyOperator->giByKey(bar); | ||||||
|                 selectedKeys.append(key); |         if (item->isCustom()) | ||||||
|  |             item->setUsedKeys(AWPatternFunctions::findKeys(item->bar(), _currentKeys, false)); | ||||||
|  |         else | ||||||
|  |             item->setUsedKeys(QStringList() << item->bar()); | ||||||
|  |         barKeys.append(item->usedKeys()); | ||||||
|     } |     } | ||||||
|         if (selectedKeys.isEmpty()) |     // get required keys | ||||||
|             qCWarning(LOG_AW) << "No bars found"; |     m_requiredKeys = m_optimize ? AWKeyCache::getRequiredKeys(m_foundKeys, barKeys, m_tooltipParams, | ||||||
|         return selectedKeys; |                                                               m_keyOperator->requiredUserKeys(), _currentKeys) | ||||||
|     }(m_pattern); |                                 : QStringList(); | ||||||
|  |  | ||||||
|     // main key list |     // set key data to m_aggregator | ||||||
|     m_foundKeys = [allKeys](QString pattern) { |     m_aggregator->setDevices(m_keyOperator->devices()); | ||||||
|         QStringList selectedKeys; |  | ||||||
|         foreach (QString key, allKeys) |  | ||||||
|             if ((!key.startsWith(QString("bar"))) |  | ||||||
|                 && (pattern.contains(QString("$%1").arg(key)))) { |  | ||||||
|                 qCInfo(LOG_AW) << "Found key" << key; |  | ||||||
|                 selectedKeys.append(key); |  | ||||||
|             } |  | ||||||
|         if (selectedKeys.isEmpty()) |  | ||||||
|             qCWarning(LOG_AW) << "No keys found"; |  | ||||||
|         return selectedKeys; |  | ||||||
|     }(m_pattern); |  | ||||||
|  |  | ||||||
|     // lambdas |  | ||||||
|     m_foundLambdas = [](QString pattern) { |  | ||||||
|         QStringList selectedKeys; |  | ||||||
|         // substring inside ${{ }} (with brackets) which should not contain ${{ |  | ||||||
|         QRegularExpression lambdaRegexp( |  | ||||||
|             QString("\\$\\{\\{((?!\\$\\{\\{).)*?\\}\\}")); |  | ||||||
|         lambdaRegexp.setPatternOptions( |  | ||||||
|             QRegularExpression::DotMatchesEverythingOption); |  | ||||||
|  |  | ||||||
|         QRegularExpressionMatchIterator it = lambdaRegexp.globalMatch(pattern); |  | ||||||
|         while (it.hasNext()) { |  | ||||||
|             QRegularExpressionMatch match = it.next(); |  | ||||||
|             QString lambda = match.captured(); |  | ||||||
|             // drop brakets |  | ||||||
|             lambda.remove(QRegExp(QString("^\\$\\{\\{"))); |  | ||||||
|             lambda.remove(QRegExp(QString("\\}\\}$"))); |  | ||||||
|             // append |  | ||||||
|             qCInfo(LOG_AW) << "Found lambda" << lambda; |  | ||||||
|             selectedKeys.append(lambda); |  | ||||||
|         } |  | ||||||
|         if (selectedKeys.isEmpty()) |  | ||||||
|             qCWarning(LOG_AW) << "No lambdas found"; |  | ||||||
|         return selectedKeys; |  | ||||||
|     }(m_pattern); |  | ||||||
|  |  | ||||||
|     // set key data to aggregator |  | ||||||
|     aggregator->setDevices(m_devices); |  | ||||||
| } | } | ||||||
|  |  | ||||||
|  |  | ||||||
| void AWKeys::updateTextData() | void AWKeys::updateTextData() | ||||||
| { | { | ||||||
| #ifdef BUILD_FUTURE |     // do not do it in parallel to avoid race condition | ||||||
|     QFuture<QString> text = QtConcurrent::run(m_threadPool, [this]() { |     m_mutex.lock(); | ||||||
|     calculateValues(); |     calculateValues(); | ||||||
|         return parsePattern(m_pattern); |     QString text = parsePattern(m_keyOperator->pattern()); | ||||||
|     }); |     // update tooltip values under lock | ||||||
| #else  /* BUILD_FUTURE */ |     m_dataAggregator->dataUpdate(m_values); | ||||||
|     calculateValues(); |     m_mutex.unlock(); | ||||||
|     QString text = parsePattern(m_pattern); |  | ||||||
| #endif /* BUILD_FUTURE */ |  | ||||||
|  |  | ||||||
|     emit(needTextToBeUpdated(text)); |     emit(needTextToBeUpdated(text)); | ||||||
|     emit(dataAggregator->updateData(values)); |  | ||||||
| } |  | ||||||
|  |  | ||||||
|  |  | ||||||
| void AWKeys::addKeyToCache(const QString type, const QString key) |  | ||||||
| { |  | ||||||
|     qCDebug(LOG_AW) << "Key type" << type; |  | ||||||
|     qCDebug(LOG_AW) << "Key" << key; |  | ||||||
|  |  | ||||||
|     QString fileName = QString("%1/awesomewidgets.ndx") |  | ||||||
|                            .arg(QStandardPaths::writableLocation( |  | ||||||
|                                QStandardPaths::GenericCacheLocation)); |  | ||||||
|     qCInfo(LOG_AW) << "Cache file" << fileName; |  | ||||||
|     QSettings cache(fileName, QSettings::IniFormat); |  | ||||||
|  |  | ||||||
|     cache.beginGroup(type); |  | ||||||
|     QStringList cachedValues; |  | ||||||
|     foreach (QString key, cache.allKeys()) |  | ||||||
|         cachedValues.append(cache.value(key).toString()); |  | ||||||
|  |  | ||||||
|     if (type == QString("hdd")) { |  | ||||||
|         QStringList allDevices |  | ||||||
|             = QDir(QString("/dev")).entryList(QDir::System, QDir::Name); |  | ||||||
|         QStringList devices |  | ||||||
|             = allDevices.filter(QRegExp(QString("^[hms]d[a-z]$"))); |  | ||||||
|         foreach (QString dev, devices) { |  | ||||||
|             QString device = QString("/dev/%1").arg(dev); |  | ||||||
|             if (cachedValues.contains(device)) |  | ||||||
|                 continue; |  | ||||||
|             qCInfo(LOG_AW) << "Found new key" << device << "for type" << type; |  | ||||||
|             cache.setValue( |  | ||||||
|                 QString("%1").arg(cache.allKeys().count(), 3, 10, QChar('0')), |  | ||||||
|                 device); |  | ||||||
|         } |  | ||||||
|     } else if (type == QString("net")) { |  | ||||||
|         QList<QNetworkInterface> rawInterfaceList |  | ||||||
|             = QNetworkInterface::allInterfaces(); |  | ||||||
|         foreach (QNetworkInterface interface, rawInterfaceList) { |  | ||||||
|             QString device = interface.name(); |  | ||||||
|             if (cachedValues.contains(device)) |  | ||||||
|                 continue; |  | ||||||
|             qCInfo(LOG_AW) << "Found new key" << device << "for type" << type; |  | ||||||
|             cache.setValue( |  | ||||||
|                 QString("%1").arg(cache.allKeys().count(), 3, 10, QChar('0')), |  | ||||||
|                 device); |  | ||||||
|         } |  | ||||||
|     } else { |  | ||||||
|         if (cachedValues.contains(key)) |  | ||||||
|             return; |  | ||||||
|         qCInfo(LOG_AW) << "Found new key" << key << "for type" << type; |  | ||||||
|         cache.setValue( |  | ||||||
|             QString("%1").arg(cache.allKeys().count(), 3, 10, QChar('0')), key); |  | ||||||
|     } |  | ||||||
|     cache.endGroup(); |  | ||||||
|  |  | ||||||
|     cache.sync(); |  | ||||||
|     return loadKeysFromCache(); |  | ||||||
| } | } | ||||||
|  |  | ||||||
|  |  | ||||||
| @ -639,151 +253,123 @@ void AWKeys::addKeyToCache(const QString type, const QString key) | |||||||
| void AWKeys::calculateValues() | void AWKeys::calculateValues() | ||||||
| { | { | ||||||
|     // hddtot* |     // hddtot* | ||||||
|     foreach (QString device, m_devices[QString("mount")]) { |     QStringList mountDevices = m_keyOperator->devices("mount"); | ||||||
|         int index = m_devices[QString("mount")].indexOf(device); |     for (auto &device : mountDevices) { | ||||||
|         values[QString("hddtotmb%1").arg(index)] = QString("%1").arg( |         int index = mountDevices.indexOf(device); | ||||||
|             values[QString("hddfreemb%1").arg(index)].toFloat() |         m_values[QString("hddtotmb%1").arg(index)] | ||||||
|                 + values[QString("hddmb%1").arg(index)].toFloat(), |             = m_values[QString("hddfreemb%1").arg(index)].toFloat() + m_values[QString("hddmb%1").arg(index)].toFloat(); | ||||||
|             5, 'f', 0); |         m_values[QString("hddtotgb%1").arg(index)] | ||||||
|         values[QString("hddtotgb%1").arg(index)] = QString("%1").arg( |             = m_values[QString("hddfreegb%1").arg(index)].toFloat() + m_values[QString("hddgb%1").arg(index)].toFloat(); | ||||||
|             values[QString("hddfreegb%1").arg(index)].toFloat() |  | ||||||
|                 + values[QString("hddgb%1").arg(index)].toFloat(), |  | ||||||
|             5, 'f', 1); |  | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     // memtot* |     // memtot* | ||||||
|     values[QString("memtotmb")] |     m_values["memtotmb"] = m_values["memusedmb"].toInt() + m_values["memfreemb"].toInt(); | ||||||
|         = QString("%1").arg(values[QString("memusedmb")].toInt() |     m_values["memtotgb"] = m_values["memusedgb"].toFloat() + m_values["memfreegb"].toFloat(); | ||||||
|                                 + values[QString("memfreemb")].toInt(), |  | ||||||
|                             5); |  | ||||||
|     values[QString("memtotgb")] |  | ||||||
|         = QString("%1").arg(values[QString("memusedgb")].toFloat() |  | ||||||
|                                 + values[QString("memfreegb")].toFloat(), |  | ||||||
|                             5, 'f', 1); |  | ||||||
|     // mem |     // mem | ||||||
|     values[QString("mem")] |     m_values["mem"] = 100.0f * m_values["memmb"].toFloat() / m_values["memtotmb"].toFloat(); | ||||||
|         = QString("%1").arg(100.0 * values[QString("memmb")].toFloat() |  | ||||||
|                                 / values[QString("memtotmb")].toFloat(), |  | ||||||
|                             5, 'f', 1); |  | ||||||
|  |  | ||||||
|     // up, down, upkb, downkb, upunits, downunits |     // up, down, upkb, downkb, upunits, downunits | ||||||
|     int netIndex = m_devices[QString("net")].indexOf(values[QString("netdev")]); |     int netIndex = m_keyOperator->devices("net").indexOf(m_values["netdev"].toString()); | ||||||
|     values[QString("down")] = values[QString("down%1").arg(netIndex)]; |     m_values["down"] = m_values[QString("down%1").arg(netIndex)]; | ||||||
|     values[QString("downkb")] = values[QString("downkb%1").arg(netIndex)]; |     m_values["downkb"] = m_values[QString("downkb%1").arg(netIndex)]; | ||||||
|     values[QString("downunits")] = values[QString("downunits%1").arg(netIndex)]; |     m_values["downtot"] = m_values[QString("downtot%1").arg(netIndex)]; | ||||||
|     values[QString("up")] = values[QString("up%1").arg(netIndex)]; |     m_values["downtotkb"] = m_values[QString("downtotkb%1").arg(netIndex)]; | ||||||
|     values[QString("upkb")] = values[QString("upkb%1").arg(netIndex)]; |     m_values["downunits"] = m_values[QString("downunits%1").arg(netIndex)]; | ||||||
|     values[QString("upunits")] = values[QString("upunits%1").arg(netIndex)]; |     m_values["up"] = m_values[QString("up%1").arg(netIndex)]; | ||||||
|  |     m_values["upkb"] = m_values[QString("upkb%1").arg(netIndex)]; | ||||||
|  |     m_values["uptot"] = m_values[QString("uptot%1").arg(netIndex)]; | ||||||
|  |     m_values["uptotkb"] = m_values[QString("uptotkb%1").arg(netIndex)]; | ||||||
|  |     m_values["upunits"] = m_values[QString("upunits%1").arg(netIndex)]; | ||||||
|  |  | ||||||
|     // swaptot* |     // swaptot* | ||||||
|     values[QString("swaptotmb")] |     m_values["swaptotmb"] = m_values["swapmb"].toInt() + m_values["swapfreemb"].toInt(); | ||||||
|         = QString("%1").arg(values[QString("swapmb")].toInt() |     m_values["swaptotgb"] = m_values["swapgb"].toFloat() + m_values["swapfreegb"].toFloat(); | ||||||
|                                 + values[QString("swapfreemb")].toInt(), |  | ||||||
|                             5); |  | ||||||
|     values[QString("swaptotgb")] |  | ||||||
|         = QString("%1").arg(values[QString("swapgb")].toFloat() |  | ||||||
|                                 + values[QString("swapfreegb")].toFloat(), |  | ||||||
|                             5, 'f', 1); |  | ||||||
|     // swap |     // swap | ||||||
|     values[QString("swap")] |     m_values["swap"] = 100.0f * m_values["swapmb"].toFloat() / m_values["swaptotmb"].toFloat(); | ||||||
|         = QString("%1").arg(100.0 * values[QString("swapmb")].toFloat() |  | ||||||
|                                 / values[QString("swaptotmb")].toFloat(), |     // user defined keys | ||||||
|                             5, 'f', 1); |     for (auto &key : m_keyOperator->userKeys()) | ||||||
|  |         m_values[key] = m_values[m_keyOperator->userKeySource(key)]; | ||||||
|  |  | ||||||
|     // lambdas |     // lambdas | ||||||
|     foreach (QString key, m_foundLambdas) |     for (auto &key : m_foundLambdas) | ||||||
|         values[key] = [this](QString key) { |         m_values[key] = AWPatternFunctions::expandLambdas(key, m_aggregator, m_values, m_foundKeys); | ||||||
|             QJSEngine engine; |  | ||||||
|             // apply $this values |  | ||||||
|             key.replace(QString("$this"), values[key]); |  | ||||||
|             foreach (QString lambdaKey, m_foundKeys) |  | ||||||
|                 key.replace(QString("$%1").arg(lambdaKey), values[lambdaKey]); |  | ||||||
|             qCInfo(LOG_AW) << "Expression" << key; |  | ||||||
|             QJSValue result = engine.evaluate(key); |  | ||||||
|             if (result.isError()) { |  | ||||||
|                 qCWarning(LOG_AW) << "Uncaught exception at line" |  | ||||||
|                                   << result.property("lineNumber").toInt() |  | ||||||
|                                   << ":" << result.toString(); |  | ||||||
|                 return QString(); |  | ||||||
|             } else { |  | ||||||
|                 return result.toString(); |  | ||||||
|             } |  | ||||||
|         }(key); |  | ||||||
| } | } | ||||||
|  |  | ||||||
|  |  | ||||||
| QString AWKeys::parsePattern(QString pattern) const | void AWKeys::createDBusInterface() | ||||||
|  | { | ||||||
|  |     // get this object id | ||||||
|  |     auto id = reinterpret_cast<qlonglong>(this); | ||||||
|  |  | ||||||
|  |     // create session | ||||||
|  |     QDBusConnection instanceBus = QDBusConnection::sessionBus(); | ||||||
|  |     // HACK we are going to use different services because it binds to | ||||||
|  |     // application | ||||||
|  |     if (instanceBus.registerService(QString("%1.i%2").arg(AWDBUS_SERVICE).arg(id))) { | ||||||
|  |         if (!instanceBus.registerObject(AWDBUS_PATH, new AWDBusAdaptor(this), QDBusConnection::ExportAllContents)) | ||||||
|  |             qCWarning(LOG_AW) << "Could not register DBus object, last error" << instanceBus.lastError().message(); | ||||||
|  |     } else { | ||||||
|  |         qCWarning(LOG_AW) << "Could not register DBus service, last error" << instanceBus.lastError().message(); | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     // and same instance but for id independent service | ||||||
|  |     QDBusConnection commonBus = QDBusConnection::sessionBus(); | ||||||
|  |     if (commonBus.registerService(AWDBUS_SERVICE)) | ||||||
|  |         commonBus.registerObject(AWDBUS_PATH, new AWDBusAdaptor(this), QDBusConnection::ExportAllContents); | ||||||
|  | } | ||||||
|  |  | ||||||
|  |  | ||||||
|  | QString AWKeys::parsePattern(QString _pattern) const | ||||||
| { | { | ||||||
|     // screen sign |     // screen sign | ||||||
|     pattern.replace(QString("$$"), QString("$\\$\\")); |     _pattern.replace("$$", QString(0x1d)); | ||||||
|  |  | ||||||
|     // lambdas |     // lambdas | ||||||
|     foreach (QString key, m_foundLambdas) |     for (auto &key : m_foundLambdas) | ||||||
|         pattern.replace(QString("${{%1}}").arg(key), values[key]); |         _pattern.replace(QString("${{%1}}").arg(key), m_values[key].toString()); | ||||||
|  |  | ||||||
|     // main keys |     // main keys | ||||||
|     foreach (QString key, m_foundKeys) |     for (auto &key : m_foundKeys) | ||||||
|         pattern.replace(QString("$%1").arg(key), |         _pattern.replace(QString("$%1").arg(key), m_aggregator->formatter(m_values[key], key, true)); | ||||||
|                         [](QString key, QString value) { |  | ||||||
|                             if ((!key.startsWith(QString("custom"))) |  | ||||||
|                                 && (!key.startsWith(QString("weather")))) |  | ||||||
|                                 value.replace(QString(" "), QString(" ")); |  | ||||||
|                             return value; |  | ||||||
|                         }(key, values[key])); |  | ||||||
|  |  | ||||||
|     // bars |     // bars | ||||||
|     foreach (QString bar, m_foundBars) { |     for (auto &bar : m_foundBars) { | ||||||
|         GraphicalItem *item = graphicalItems->itemByTag(bar); |         GraphicalItem *item = m_keyOperator->giByKey(bar); | ||||||
|         QString key = bar; |         QString image = item->isCustom() ? item->image( | ||||||
|         key.remove(QRegExp(QString("^bar[0-9]{1,}"))); |                             AWPatternFunctions::expandLambdas(item->bar(), m_aggregator, m_values, item->usedKeys())) | ||||||
|         if (item->type() == GraphicalItem::Graph) |                                          : item->image(m_values[item->bar()]); | ||||||
|             pattern.replace(QString("$%1").arg(bar), |         _pattern.replace(QString("$%1").arg(bar), image); | ||||||
|                             item->image([](const QList<float> data) { |  | ||||||
|                                 return QVariant::fromValue<QList<float>>(data); |  | ||||||
|                             }(dataAggregator->getData(key)))); |  | ||||||
|         else |  | ||||||
|             pattern.replace(QString("$%1").arg(bar), item->image(values[key])); |  | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     // prepare strings |     // prepare strings | ||||||
|     pattern.replace(QString("$\\$\\"), QString("$$")); |     _pattern.replace(QString(0x1d), "$"); | ||||||
|     if (m_wrapNewLines) |     if (m_wrapNewLines) | ||||||
|         pattern.replace(QString("\n"), QString("<br>")); |         _pattern.replace("\n", "<br>"); | ||||||
|  |  | ||||||
|     return pattern; |     return _pattern; | ||||||
| } | } | ||||||
|  |  | ||||||
|  |  | ||||||
| void AWKeys::setDataBySource(const QString &sourceName, const QVariantMap &data) | void AWKeys::setDataBySource(const QString &_sourceName, const QVariantMap &_data) | ||||||
| { | { | ||||||
|     qCDebug(LOG_AW) << "Source" << sourceName; |     qCDebug(LOG_AW) << "Source" << _sourceName << "with data" << _data; | ||||||
|     qCDebug(LOG_AW) << "Data" << data; |  | ||||||
|  |  | ||||||
|     // first list init |     // first list init | ||||||
|     QStringList tags = aggregator->keysFromSource(sourceName); |     QStringList tags = m_aggregator->keysFromSource(_sourceName); | ||||||
|     if (tags.isEmpty()) |     if (tags.isEmpty()) | ||||||
|         tags = aggregator->registerSource(sourceName, |         tags = m_aggregator->registerSource(_sourceName, _data["units"].toString(), m_requiredKeys); | ||||||
|                                           data[QString("units")].toString()); |  | ||||||
|  |  | ||||||
|     // update data or drop source if there are no matches |     // update data or drop source if there are no matches and exit | ||||||
|     if (tags.isEmpty()) { |     if (tags.isEmpty()) { | ||||||
|         qCDebug(LOG_AW) << "Source" << sourceName << "not found"; |         qCInfo(LOG_AW) << "Source" << _sourceName << "not found"; | ||||||
|         emit(dropSourceFromDataengine(sourceName)); |         return emit(dropSourceFromDataengine(_sourceName)); | ||||||
|     } else { |  | ||||||
| #ifdef BUILD_FUTURE |  | ||||||
|         m_mutex.lock(); |  | ||||||
| #endif /* BUILD_FUTURE */ |  | ||||||
|         // HACK workaround for time values which are stored in the different |  | ||||||
|         // path |  | ||||||
|         QVariant value = sourceName == QString("Local") |  | ||||||
|                              ? data[QString("DateTime")] |  | ||||||
|                              : data[QString("value")]; |  | ||||||
|         std::for_each(tags.cbegin(), tags.cend(), |  | ||||||
|                       [this, value](const QString tag) { |  | ||||||
|                           values[tag] = aggregator->formater(value, tag); |  | ||||||
|                       }); |  | ||||||
| #ifdef BUILD_FUTURE |  | ||||||
|         m_mutex.unlock(); |  | ||||||
| #endif /* BUILD_FUTURE */ |  | ||||||
|     } |     } | ||||||
|  |  | ||||||
|  |     m_mutex.lock(); | ||||||
|  |     // HACK workaround for time values which are stored in the different path | ||||||
|  |     std::for_each(tags.cbegin(), tags.cend(), [this, &_data, &_sourceName](const QString &tag) { | ||||||
|  |         m_values[tag] = _sourceName == "Local" ? _data["DateTime"] : _data["value"]; | ||||||
|  |     }); | ||||||
|  |     m_mutex.unlock(); | ||||||
| } | } | ||||||
|  | |||||||
| @ -24,81 +24,68 @@ | |||||||
| #include <QMutex> | #include <QMutex> | ||||||
| #include <QObject> | #include <QObject> | ||||||
|  |  | ||||||
| #include "extitemaggregator.h" |  | ||||||
|  |  | ||||||
|  |  | ||||||
| class AWDataAggregator; | class AWDataAggregator; | ||||||
| class AWDataEngineAggregator; | class AWDataEngineAggregator; | ||||||
|  | class AWKeyOperations; | ||||||
| class AWKeysAggregator; | class AWKeysAggregator; | ||||||
| class ExtQuotes; |  | ||||||
| class ExtScript; |  | ||||||
| class ExtUpgrade; |  | ||||||
| class ExtWeather; |  | ||||||
| class GraphicalItem; |  | ||||||
| class QThreadPool; | class QThreadPool; | ||||||
|  | class QTimer; | ||||||
|  |  | ||||||
| class AWKeys : public QObject | class AWKeys : public QObject | ||||||
| { | { | ||||||
|     Q_OBJECT |     Q_OBJECT | ||||||
|  |  | ||||||
| public: | public: | ||||||
|     explicit AWKeys(QObject *parent = nullptr); |     explicit AWKeys(QObject *_parent = nullptr); | ||||||
|     virtual ~AWKeys(); |     ~AWKeys() override; | ||||||
|     Q_INVOKABLE void initDataAggregator(const QVariantMap tooltipParams); |     Q_INVOKABLE void initDataAggregator(const QVariantMap &_tooltipParams); | ||||||
|     Q_INVOKABLE void initKeys(const QString currentPattern, const int interval, |     Q_INVOKABLE void initKeys(const QString &_currentPattern, int _interval, int _limit, bool _optimize); | ||||||
|                               const int limit); |     Q_INVOKABLE void setAggregatorProperty(const QString &_key, const QVariant &_value); | ||||||
|     Q_INVOKABLE void setAggregatorProperty(const QString key, |     Q_INVOKABLE void setWrapNewLines(bool _wrap); | ||||||
|                                            const QVariant value); |     // additional method to force load keys from Qml UI. Used in some | ||||||
|     Q_INVOKABLE void setWrapNewLines(const bool wrap = false); |     // configuration pages | ||||||
|     Q_INVOKABLE void updateCache(); |     Q_INVOKABLE void updateCache(); | ||||||
|     // keys |     // keys | ||||||
|     Q_INVOKABLE QStringList dictKeys(const bool sorted = false, |     Q_INVOKABLE [[nodiscard]] QStringList dictKeys(bool _sorted = false, const QString &_regexp = "") const; | ||||||
|                                      const QString regexp = QString()) const; |     Q_INVOKABLE [[nodiscard]] QVariantList getHddDevices() const; | ||||||
|     Q_INVOKABLE QStringList getHddDevices() const; |  | ||||||
|     // values |     // values | ||||||
|     Q_INVOKABLE QString infoByKey(QString key) const; |     Q_INVOKABLE [[nodiscard]] QString infoByKey(const QString &_key) const; | ||||||
|     Q_INVOKABLE QString valueByKey(QString key) const; |     Q_INVOKABLE [[nodiscard]] QString valueByKey(const QString &_key) const; | ||||||
|     // configuration |     // configuration | ||||||
|     Q_INVOKABLE void editItem(const QString type); |     Q_INVOKABLE void editItem(const QString &_type); | ||||||
|  |  | ||||||
| public slots: | public slots: | ||||||
|     void addDevice(const QString source); |     void dataUpdated(const QString &_sourceName, const Plasma::DataEngine::Data &_data); | ||||||
|     void dataUpdated(const QString &sourceName, |  | ||||||
|                      const Plasma::DataEngine::Data &data); |  | ||||||
|     // dummy method required by DataEngine connections |     // dummy method required by DataEngine connections | ||||||
|     void modelChanged(QString, QAbstractItemModel *){}; |     static void modelChanged(const QString &, QAbstractItemModel *){}; | ||||||
|  |  | ||||||
| signals: | signals: | ||||||
|     void dropSourceFromDataengine(const QString source); |     void dropSourceFromDataengine(const QString &_source); | ||||||
|     void needTextToBeUpdated(const QString newText) const; |     void needTextToBeUpdated(const QString &_newText) const; | ||||||
|     void needToolTipToBeUpdated(const QString newText) const; |     void needToolTipToBeUpdated(const QString &_newText) const; | ||||||
|     void needToBeUpdated(); |  | ||||||
|  |  | ||||||
| private slots: | private slots: | ||||||
|     void loadKeysFromCache(); |     void reinitKeys(const QStringList &_currentKeys); | ||||||
|     void reinitKeys(); |  | ||||||
|     void updateTextData(); |     void updateTextData(); | ||||||
|  |  | ||||||
| private: | private: | ||||||
|     // methods |     // methods | ||||||
|     void addKeyToCache(const QString type, const QString key = QString("")); |  | ||||||
|     void calculateValues(); |     void calculateValues(); | ||||||
|     QString parsePattern(QString pattern) const; |     void createDBusInterface(); | ||||||
|     void setDataBySource(const QString &sourceName, const QVariantMap &data); |     [[nodiscard]] QString parsePattern(QString _pattern) const; | ||||||
|  |     void setDataBySource(const QString &_sourceName, const QVariantMap &_data); | ||||||
|     // objects |     // objects | ||||||
|     AWDataAggregator *dataAggregator = nullptr; |     AWDataAggregator *m_dataAggregator = nullptr; | ||||||
|     AWDataEngineAggregator *dataEngineAggregator = nullptr; |     AWDataEngineAggregator *m_dataEngineAggregator = nullptr; | ||||||
|     AWKeysAggregator *aggregator = nullptr; |     AWKeysAggregator *m_aggregator = nullptr; | ||||||
|     ExtItemAggregator<GraphicalItem> *graphicalItems = nullptr; |     AWKeyOperations *m_keyOperator = nullptr; | ||||||
|     ExtItemAggregator<ExtQuotes> *extQuotes = nullptr; |     QTimer *m_timer = nullptr; | ||||||
|     ExtItemAggregator<ExtScript> *extScripts = nullptr; |  | ||||||
|     ExtItemAggregator<ExtUpgrade> *extUpgrade = nullptr; |  | ||||||
|     ExtItemAggregator<ExtWeather> *extWeather = nullptr; |  | ||||||
|     // variables |     // variables | ||||||
|     QHash<QString, QStringList> m_devices; |     QVariantMap m_tooltipParams; | ||||||
|     QStringList m_foundBars, m_foundKeys, m_foundLambdas; |     QStringList m_foundBars, m_foundKeys, m_foundLambdas, m_requiredKeys; | ||||||
|     QString m_pattern; |     QVariantHash m_values; | ||||||
|     QHash<QString, QString> values; |     bool m_optimize = false; | ||||||
|     bool m_wrapNewLines = false; |     bool m_wrapNewLines = false; | ||||||
|     // multithread features |     // multithread features | ||||||
|     QThreadPool *m_threadPool = nullptr; |     QThreadPool *m_threadPool = nullptr; | ||||||
|  | |||||||
| @ -21,15 +21,25 @@ | |||||||
|  |  | ||||||
| #include <QDateTime> | #include <QDateTime> | ||||||
| #include <QLocale> | #include <QLocale> | ||||||
| #include <QRegExp> |  | ||||||
|  |  | ||||||
|  | #include "awdataenginemapper.h" | ||||||
| #include "awdebug.h" | #include "awdebug.h" | ||||||
|  | #include "awformatterhelper.h" | ||||||
|  | #include "version.h" | ||||||
|  |  | ||||||
|  |  | ||||||
| AWKeysAggregator::AWKeysAggregator(QObject *parent) | AWKeysAggregator::AWKeysAggregator(QObject *_parent) | ||||||
|     : QObject(parent) |     : QObject(_parent) | ||||||
| { | { | ||||||
|     qCDebug(LOG_AW) << __PRETTY_FUNCTION__; |     qCDebug(LOG_AW) << __PRETTY_FUNCTION__; | ||||||
|  |  | ||||||
|  |     m_customFormatters = new AWFormatterHelper(nullptr); | ||||||
|  |     m_mapper = new AWDataEngineMapper(this, m_customFormatters); | ||||||
|  |  | ||||||
|  |     // sort time keys | ||||||
|  |     m_timeKeys = QString(TIME_KEYS).split(','); | ||||||
|  |     m_timeKeys.sort(); | ||||||
|  |     std::reverse(m_timeKeys.begin(), m_timeKeys.end()); | ||||||
| } | } | ||||||
|  |  | ||||||
|  |  | ||||||
| @ -39,480 +49,196 @@ AWKeysAggregator::~AWKeysAggregator() | |||||||
| } | } | ||||||
|  |  | ||||||
|  |  | ||||||
| QString AWKeysAggregator::formater(const QVariant &data, | void AWKeysAggregator::initFormatters() | ||||||
|                                    const QString &key) const |  | ||||||
| { | { | ||||||
|     qCDebug(LOG_AW) << "Data" << data; |     m_customFormatters->initItems(); | ||||||
|     qCDebug(LOG_AW) << "Key" << key; | } | ||||||
|  |  | ||||||
|  |  | ||||||
|  | QString AWKeysAggregator::formatter(const QVariant &_data, const QString &_key, bool replaceSpace) const | ||||||
|  | { | ||||||
|  |     qCDebug(LOG_AW) << "Data" << _data << "for key" << _key; | ||||||
|  |  | ||||||
|     QString output; |     QString output; | ||||||
|     QLocale loc = m_translate ? QLocale::system() : QLocale::c(); |     QLocale loc = m_translate ? QLocale::system() : QLocale::c(); | ||||||
|     // case block |     // case block | ||||||
|     switch (m_formater[key]) { |     switch (m_mapper->formatter(_key)) { | ||||||
|     case Float: |     case FormatterType::Float: | ||||||
|         output = QString("%1").arg(data.toFloat(), 5, 'f', 1); |         output = QString("%1").arg(_data.toFloat(), 5, 'f', 1); | ||||||
|         break; |         break; | ||||||
|     case FloatTwoSymbols: |     case FormatterType::FloatTwoSymbols: | ||||||
|         output = QString("%1").arg(data.toFloat(), 5, 'f', 2); |         output = QString("%1").arg(_data.toFloat(), 5, 'f', 2); | ||||||
|         break; |         break; | ||||||
|     case Integer: |     case FormatterType::Integer: | ||||||
|         output = QString("%1").arg(data.toFloat(), 4, 'f', 0); |         output = QString("%1").arg(_data.toFloat(), 4, 'f', 0); | ||||||
|         break; |         break; | ||||||
|     case IntegerThree: |     case FormatterType::IntegerFive: | ||||||
|         output = QString("%1").arg(data.toFloat(), 3, 'f', 0); |         output = QString("%1").arg(_data.toFloat(), 5, 'f', 0); | ||||||
|         break; |         break; | ||||||
|     case List: |     case FormatterType::IntegerThree: | ||||||
|         output = data.toStringList().join(QChar(',')); |         output = QString("%1").arg(_data.toFloat(), 3, 'f', 0); | ||||||
|         break; |         break; | ||||||
|     case ACFormat: |     case FormatterType::List: | ||||||
|         output = data.toBool() ? m_acOnline : m_acOffline; |         output = _data.toStringList().join(','); | ||||||
|         break; |         break; | ||||||
|     case MemGBFormat: |     case FormatterType::ACFormat: | ||||||
|         output |         output = _data.toBool() ? m_acOnline : m_acOffline; | ||||||
|             = QString("%1").arg(data.toFloat() / (1024.0 * 1024.0), 5, 'f', 1); |  | ||||||
|         break; |         break; | ||||||
|     case MemMBFormat: |     case FormatterType::MemGBFormat: | ||||||
|         output = QString("%1").arg(data.toFloat() / 1024.0, 5, 'f', 0); |         output = QString("%1").arg(_data.toFloat() / (1024.0 * 1024.0), 5, 'f', 1); | ||||||
|         break; |         break; | ||||||
|     case NetSmartFormat: |     case FormatterType::MemMBFormat: | ||||||
|  |         output = QString("%1").arg(_data.toFloat() / 1024.0, 5, 'f', 0); | ||||||
|  |         break; | ||||||
|  |     case FormatterType::NetSmartFormat: | ||||||
|         output = [](const float value) { |         output = [](const float value) { | ||||||
|             if (value > 1024.0) |             if (value > 1024.0) | ||||||
|                 return QString("%1").arg(value / 1024.0, 4, 'f', 1); |                 return QString("%1").arg(value / 1024.0, 4, 'f', 1); | ||||||
|             else |             else | ||||||
|                 return QString("%1").arg(value, 4, 'f', 0); |                 return QString("%1").arg(value, 4, 'f', 0); | ||||||
|         }(data.toFloat()); |         }(_data.toFloat()); | ||||||
|         break; |         break; | ||||||
|     case NetSmartUnits: |     case FormatterType::NetSmartUnits: | ||||||
|         if (data.toFloat() > 1024.0) |         if (_data.toFloat() > 1024.0) | ||||||
|             output = m_translate ? i18n("MB/s") : QString("MB/s"); |             output = m_translate ? i18n("MB/s") : "MB/s"; | ||||||
|         else |         else | ||||||
|             output = m_translate ? i18n("KB/s") : QString("KB/s"); |             output = m_translate ? i18n("KB/s") : "KB/s"; | ||||||
|         break; |         break; | ||||||
|     case Quotes: |     case FormatterType::Quotes: | ||||||
|         // first cast |         // first cast | ||||||
|         output = QString("%1").arg(data.toDouble(), 0, 'f'); |         output = QString("%1").arg(_data.toDouble(), 0, 'f'); | ||||||
|         output = output.rightJustified(8, QLatin1Char(' '), true); |         output = output.rightJustified(8, QLatin1Char(' '), true); | ||||||
|         break; |         break; | ||||||
|     case Temperature: |     case FormatterType::Temperature: | ||||||
|         output = QString("%1").arg(temperature(data.toFloat()), 5, 'f', 1); |         output = QString("%1").arg(temperature(_data.toFloat()), 5, 'f', 1); | ||||||
|         break; |         break; | ||||||
|     case Time: |     case FormatterType::Time: | ||||||
|         output = data.toDateTime().toString(); |         output = _data.toDateTime().toString(); | ||||||
|         break; |         break; | ||||||
|     case TimeCustom: |     case FormatterType::TimeCustom: | ||||||
|         output = m_customTime; |         output = m_customTime; | ||||||
|         [&output, loc, this](const QDateTime dt) { |         [&output, loc, this](const QDateTime &dt) { | ||||||
|             foreach (QString key, timeKeys) |             for (auto &key : m_timeKeys) | ||||||
|                 output.replace(QString("$%1").arg(key), loc.toString(dt, key)); |                 output.replace(QString("$%1").arg(key), loc.toString(dt, key)); | ||||||
|         }(data.toDateTime()); |         }(_data.toDateTime()); | ||||||
|         break; |         break; | ||||||
|     case TimeISO: |     case FormatterType::TimeISO: | ||||||
|         output = data.toDateTime().toString(Qt::ISODate); |         output = _data.toDateTime().toString(Qt::ISODate); | ||||||
|         break; |         break; | ||||||
|     case TimeLong: |     case FormatterType::TimeLong: | ||||||
|         output = loc.toString(data.toDateTime(), QLocale::LongFormat); |         output = loc.toString(_data.toDateTime(), QLocale::LongFormat); | ||||||
|         break; |         break; | ||||||
|     case TimeShort: |     case FormatterType::TimeShort: | ||||||
|         output = loc.toString(data.toDateTime(), QLocale::ShortFormat); |         output = loc.toString(_data.toDateTime(), QLocale::ShortFormat); | ||||||
|         break; |         break; | ||||||
|     case Uptime: |     case FormatterType::Timestamp: | ||||||
|     case UptimeCustom: |         output = QString("%1").arg(_data.toDateTime().toMSecsSinceEpoch() / 1000.0, 10, 'f', 0); | ||||||
|  |         break; | ||||||
|  |     case FormatterType::Uptime: | ||||||
|  |     case FormatterType::UptimeCustom: | ||||||
|         output = |         output = | ||||||
|             [](QString source, const int uptime) { |             [](QString source, const int uptime) { | ||||||
|                 int seconds = uptime - uptime % 60; |                 int seconds = uptime - uptime % 60; | ||||||
|                 int minutes = seconds / 60 % 60; |                 int minutes = seconds / 60 % 60; | ||||||
|                 int hours = ((seconds / 60) - minutes) / 60 % 24; |                 int hours = ((seconds / 60) - minutes) / 60 % 24; | ||||||
|                 int days = (((seconds / 60) - minutes) / 60 - hours) / 24; |                 int days = (((seconds / 60) - minutes) / 60 - hours) / 24; | ||||||
|                 source.replace(QString("$dd"), |                 source.replace("$dd", QString("%1").arg(days, 3, 10, QChar('0'))); | ||||||
|                                QString("%1").arg(days, 3, 10, QChar('0'))); |                 source.replace("$d", QString("%1").arg(days)); | ||||||
|                 source.replace(QString("$d"), QString("%1").arg(days)); |                 source.replace("$hh", QString("%1").arg(hours, 2, 10, QChar('0'))); | ||||||
|                 source.replace(QString("$hh"), |                 source.replace("$h", QString("%1").arg(hours)); | ||||||
|                                QString("%1").arg(hours, 2, 10, QChar('0'))); |                 source.replace("$mm", QString("%1").arg(minutes, 2, 10, QChar('0'))); | ||||||
|                 source.replace(QString("$h"), QString("%1").arg(hours)); |                 source.replace("$m", QString("%1").arg(minutes)); | ||||||
|                 source.replace(QString("$mm"), |  | ||||||
|                                QString("%1").arg(minutes, 2, 10, QChar('0'))); |  | ||||||
|                 source.replace(QString("$m"), QString("%1").arg(minutes)); |  | ||||||
|                 return source; |                 return source; | ||||||
|             }(m_formater[key] == Uptime ? QString("$ddd$hhh$mmm") |             }(m_mapper->formatter(_key) == FormatterType::Uptime ? "$ddd$hhh$mmm" : m_customUptime, | ||||||
|                                         : m_customUptime, |               static_cast<int>(_data.toFloat())); | ||||||
|               data.toFloat()); |  | ||||||
|         break; |         break; | ||||||
|     case NoFormat: |     case FormatterType::NoFormat: | ||||||
|     default: |         output = _data.toString(); | ||||||
|         output = data.toString(); |         break; | ||||||
|  |     case FormatterType::Custom: | ||||||
|  |         if (m_customFormatters) | ||||||
|  |             output = m_customFormatters->convert(_data, _key); | ||||||
|         break; |         break; | ||||||
|     } |     } | ||||||
|  |  | ||||||
|  |     // replace spaces to non-breakable ones | ||||||
|  |     replaceSpace &= (!_key.startsWith("custom") && (!_key.startsWith("weather"))); | ||||||
|  |     if (replaceSpace) | ||||||
|  |         output.replace(" ", " "); | ||||||
|  |  | ||||||
|     return output; |     return output; | ||||||
| } | } | ||||||
|  |  | ||||||
|  |  | ||||||
| QStringList AWKeysAggregator::keysFromSource(const QString &source) const | QStringList AWKeysAggregator::keysFromSource(const QString &_source) const | ||||||
| { | { | ||||||
|     qCDebug(LOG_AW) << "Search for source" << source; |     qCDebug(LOG_AW) << "Search for source" << _source; | ||||||
|  |  | ||||||
|     return m_map.values(source); |     return m_mapper->keysFromSource(_source); | ||||||
| } | } | ||||||
|  |  | ||||||
|  |  | ||||||
| void AWKeysAggregator::setAcOffline(const QString inactive) | void AWKeysAggregator::setAcOffline(const QString &_inactive) | ||||||
| { | { | ||||||
|     qCDebug(LOG_AW) << "Inactive AC string" << inactive; |     qCDebug(LOG_AW) << "Inactive AC string" << _inactive; | ||||||
|  |  | ||||||
|     m_acOffline = inactive; |     m_acOffline = _inactive; | ||||||
| } | } | ||||||
|  |  | ||||||
|  |  | ||||||
| void AWKeysAggregator::setAcOnline(const QString active) | void AWKeysAggregator::setAcOnline(const QString &_active) | ||||||
| { | { | ||||||
|     qCDebug(LOG_AW) << "Active AC string" << active; |     qCDebug(LOG_AW) << "Active AC string" << _active; | ||||||
|  |  | ||||||
|     m_acOnline = active; |     m_acOnline = _active; | ||||||
| } | } | ||||||
|  |  | ||||||
|  |  | ||||||
| void AWKeysAggregator::setCustomTime(const QString customTime) | void AWKeysAggregator::setCustomTime(const QString &_customTime) | ||||||
| { | { | ||||||
|     qCDebug(LOG_AW) << "Format" << customTime; |     qCDebug(LOG_AW) << "Format" << _customTime; | ||||||
|  |  | ||||||
|     m_customTime = customTime; |     m_customTime = _customTime; | ||||||
| } | } | ||||||
|  |  | ||||||
|  |  | ||||||
| void AWKeysAggregator::setCustomUptime(const QString customUptime) | void AWKeysAggregator::setCustomUptime(const QString &_customUptime) | ||||||
| { | { | ||||||
|     qCDebug(LOG_AW) << "Format" << customUptime; |     qCDebug(LOG_AW) << "Format" << _customUptime; | ||||||
|  |  | ||||||
|     m_customUptime = customUptime; |     m_customUptime = _customUptime; | ||||||
| } | } | ||||||
|  |  | ||||||
|  |  | ||||||
| void AWKeysAggregator::setDevices(const QHash<QString, QStringList> devices) | void AWKeysAggregator::setDevices(const QHash<QString, QStringList> &_devices) | ||||||
| { | { | ||||||
|     qCDebug(LOG_AW) << "Devices" << devices; |     qCDebug(LOG_AW) << "Devices" << _devices; | ||||||
|  |  | ||||||
|     m_devices = devices; |     m_mapper->setDevices(_devices); | ||||||
| } | } | ||||||
|  |  | ||||||
|  |  | ||||||
| void AWKeysAggregator::setTempUnits(const QString units) | void AWKeysAggregator::setTempUnits(const QString &_units) | ||||||
| { | { | ||||||
|     qCDebug(LOG_AW) << "Units" << units; |     qCDebug(LOG_AW) << "Units" << _units; | ||||||
|  |  | ||||||
|     m_tempUnits = units; |     m_tempUnits = _units; | ||||||
| } | } | ||||||
|  |  | ||||||
|  |  | ||||||
| void AWKeysAggregator::setTranslate(const bool translate) | void AWKeysAggregator::setTranslate(const bool _translate) | ||||||
| { | { | ||||||
|     qCDebug(LOG_AW) << "Translate" << translate; |     qCDebug(LOG_AW) << "Translate" << _translate; | ||||||
|  |  | ||||||
|     m_translate = translate; |     m_translate = _translate; | ||||||
| } | } | ||||||
|  |  | ||||||
|  |  | ||||||
| // HACK units required to define should the value be calculated as temperature | QStringList AWKeysAggregator::registerSource(const QString &_source, const QString &_units, const QStringList &_keys) | ||||||
| // or fan data |  | ||||||
| QStringList AWKeysAggregator::registerSource(const QString &source, |  | ||||||
|                                              const QString &units) |  | ||||||
| { | { | ||||||
|     qCDebug(LOG_AW) << "Source" << source; |     qCDebug(LOG_AW) << "Source" << _source << "with units" << _units; | ||||||
|     qCDebug(LOG_AW) << "Units" << units; |  | ||||||
|  |  | ||||||
|     // regular expressions |     return m_mapper->registerSource(_source, _units, _keys); | ||||||
|     QRegExp cpuRegExp = QRegExp(QString("cpu/cpu.*/TotalLoad")); |  | ||||||
|     QRegExp cpuclRegExp = QRegExp(QString("cpu/cpu.*/clock")); |  | ||||||
|     QRegExp hddrRegExp = QRegExp(QString("disk/.*/Rate/rblk")); |  | ||||||
|     QRegExp hddwRegExp = QRegExp(QString("disk/.*/Rate/wblk")); |  | ||||||
|     QRegExp mountFillRegExp = QRegExp(QString("partitions/.*/filllevel")); |  | ||||||
|     QRegExp mountFreeRegExp = QRegExp(QString("partitions/.*/freespace")); |  | ||||||
|     QRegExp mountUsedRegExp = QRegExp(QString("partitions/.*/usedspace")); |  | ||||||
|     QRegExp netRegExp = QRegExp( |  | ||||||
|         QString("network/interfaces/.*/(receiver|transmitter)/data$")); |  | ||||||
|  |  | ||||||
|     if (source == QString("battery/ac")) { |  | ||||||
|         // AC |  | ||||||
|         m_map[source] = QString("ac"); |  | ||||||
|         m_formater[QString("ac")] = ACFormat; |  | ||||||
|     } else if (source.startsWith(QString("battery/"))) { |  | ||||||
|         // battery stats |  | ||||||
|         QString key = source; |  | ||||||
|         key.remove(QString("battery/")); |  | ||||||
|         m_map[source] = key; |  | ||||||
|         m_formater[key] = IntegerThree; |  | ||||||
|     } else if (source == QString("cpu/system/TotalLoad")) { |  | ||||||
|         // cpu |  | ||||||
|         m_map[source] = QString("cpu"); |  | ||||||
|         m_formater[QString("cpu")] = Float; |  | ||||||
|     } else if (source.contains(cpuRegExp)) { |  | ||||||
|         // cpus |  | ||||||
|         QString key = source; |  | ||||||
|         key.remove(QString("cpu/")).remove(QString("/TotalLoad")); |  | ||||||
|         m_map[source] = key; |  | ||||||
|         m_formater[key] = Float; |  | ||||||
|     } else if (source == QString("cpu/system/AverageClock")) { |  | ||||||
|         // cpucl |  | ||||||
|         m_map[source] = QString("cpucl"); |  | ||||||
|         m_formater[QString("cpucl")] = Integer; |  | ||||||
|     } else if (source.contains(cpuclRegExp)) { |  | ||||||
|         // cpucls |  | ||||||
|         QString key = source; |  | ||||||
|         key.remove(QString("cpu/cpu")).remove(QString("/clock")); |  | ||||||
|         key = QString("cpucl%1").arg(key); |  | ||||||
|         m_map[source] = key; |  | ||||||
|         m_formater[key] = Integer; |  | ||||||
|     } else if (source.startsWith(QString("custom"))) { |  | ||||||
|         // custom |  | ||||||
|         QString key = source; |  | ||||||
|         key.remove(QString("custom/")); |  | ||||||
|         m_map[source] = key; |  | ||||||
|         m_formater[key] = NoFormat; |  | ||||||
|     } else if (source == QString("desktop/current/name")) { |  | ||||||
|         // current desktop name |  | ||||||
|         m_map[source] = QString("desktop"); |  | ||||||
|         m_formater[QString("desktop")] = NoFormat; |  | ||||||
|     } else if (source == QString("desktop/current/number")) { |  | ||||||
|         // current desktop number |  | ||||||
|         m_map[source] = QString("ndesktop"); |  | ||||||
|         m_formater[QString("ndesktop")] = NoFormat; |  | ||||||
|     } else if (source == QString("desktop/total/number")) { |  | ||||||
|         // desktop count |  | ||||||
|         m_map[source] = QString("tdesktops"); |  | ||||||
|         m_formater[QString("tdesktops")] = NoFormat; |  | ||||||
|     } else if (source.contains(hddrRegExp)) { |  | ||||||
|         // read speed |  | ||||||
|         QString device = source; |  | ||||||
|         device.remove(QString("/Rate/rblk")); |  | ||||||
|         int index = m_devices[QString("disk")].indexOf(device); |  | ||||||
|         if (index > -1) { |  | ||||||
|             QString key = QString("hddr%1").arg(index); |  | ||||||
|             m_map[source] = key; |  | ||||||
|             m_formater[key] = Integer; |  | ||||||
|         } |  | ||||||
|     } else if (source.contains(hddwRegExp)) { |  | ||||||
|         // write speed |  | ||||||
|         QString device = source; |  | ||||||
|         device.remove(QString("/Rate/wblk")); |  | ||||||
|         int index = m_devices[QString("disk")].indexOf(device); |  | ||||||
|         if (index > -1) { |  | ||||||
|             QString key = QString("hddw%1").arg(index); |  | ||||||
|             m_map[source] = key; |  | ||||||
|             m_formater[key] = Integer; |  | ||||||
|         } |  | ||||||
|     } else if (source == QString("gpu/load")) { |  | ||||||
|         // gpu load |  | ||||||
|         m_map[source] = QString("gpu"); |  | ||||||
|         m_formater[QString("gpu")] = Float; |  | ||||||
|     } else if (source == QString("gpu/temperature")) { |  | ||||||
|         // gpu temperature |  | ||||||
|         m_map[source] = QString("gputemp"); |  | ||||||
|         m_formater[QString("gputemp")] = Temperature; |  | ||||||
|     } else if (source.contains(mountFillRegExp)) { |  | ||||||
|         // fill level |  | ||||||
|         QString device = source; |  | ||||||
|         device.remove(QString("partitions")).remove(QString("/filllevel")); |  | ||||||
|         int index = m_devices[QString("mount")].indexOf(device); |  | ||||||
|         if (index > -1) { |  | ||||||
|             QString key = QString("hdd%1").arg(index); |  | ||||||
|             m_map[source] = key; |  | ||||||
|             m_formater[key] = Float; |  | ||||||
|         } |  | ||||||
|     } else if (source.contains(mountFreeRegExp)) { |  | ||||||
|         // free space |  | ||||||
|         QString device = source; |  | ||||||
|         device.remove(QString("partitions")).remove(QString("/freespace")); |  | ||||||
|         int index = m_devices[QString("mount")].indexOf(device); |  | ||||||
|         if (index > -1) { |  | ||||||
|             // mb |  | ||||||
|             QString key = QString("hddfreemb%1").arg(index); |  | ||||||
|             m_map[source] = key; |  | ||||||
|             m_formater[key] = MemMBFormat; |  | ||||||
|             // gb |  | ||||||
|             key = QString("hddfreegb%1").arg(index); |  | ||||||
|             m_map.insertMulti(source, key); |  | ||||||
|             m_formater[key] = MemGBFormat; |  | ||||||
|         } |  | ||||||
|     } else if (source.contains(mountUsedRegExp)) { |  | ||||||
|         // used |  | ||||||
|         QString device = source; |  | ||||||
|         device.remove(QString("partitions")).remove(QString("/usedspace")); |  | ||||||
|         int index = m_devices[QString("mount")].indexOf(device); |  | ||||||
|         if (index > -1) { |  | ||||||
|             // mb |  | ||||||
|             QString key = QString("hddmb%1").arg(index); |  | ||||||
|             m_map[source] = key; |  | ||||||
|             m_formater[key] = MemMBFormat; |  | ||||||
|             // gb |  | ||||||
|             key = QString("hddgb%1").arg(index); |  | ||||||
|             m_map.insertMulti(source, key); |  | ||||||
|             m_formater[key] = MemGBFormat; |  | ||||||
|         } |  | ||||||
|     } else if (source.startsWith(QString("hdd/temperature"))) { |  | ||||||
|         // hdd temperature |  | ||||||
|         QString device = source; |  | ||||||
|         device.remove(QString("hdd/temperature")); |  | ||||||
|         int index = m_devices[QString("hdd")].indexOf(device); |  | ||||||
|         if (index > -1) { |  | ||||||
|             QString key = QString("hddtemp%1").arg(index); |  | ||||||
|             m_map[source] = key; |  | ||||||
|             m_formater[key] = Temperature; |  | ||||||
|         } |  | ||||||
|     } else if (source.startsWith(QString("cpu/system/loadavg"))) { |  | ||||||
|         // load average |  | ||||||
|         QString time = source; |  | ||||||
|         time.remove(QString("cpu/system/loadavg")); |  | ||||||
|         QString key = QString("la%1").arg(time); |  | ||||||
|         m_map[source] = key; |  | ||||||
|         m_formater[key] = FloatTwoSymbols; |  | ||||||
|     } else if (source == QString("mem/physical/application")) { |  | ||||||
|         // app memory |  | ||||||
|         // mb |  | ||||||
|         m_map[source] = QString("memmb"); |  | ||||||
|         m_formater[QString("memmb")] = MemMBFormat; |  | ||||||
|         // gb |  | ||||||
|         m_map.insertMulti(source, QString("memgb")); |  | ||||||
|         m_formater[QString("memgb")] = MemGBFormat; |  | ||||||
|     } else if (source == QString("mem/physical/free")) { |  | ||||||
|         // free memory |  | ||||||
|         // mb |  | ||||||
|         m_map[source] = QString("memfreemb"); |  | ||||||
|         m_formater[QString("memfreemb")] = MemMBFormat; |  | ||||||
|         // gb |  | ||||||
|         m_map.insertMulti(source, QString("memfreegb")); |  | ||||||
|         m_formater[QString("memfreegb")] = MemGBFormat; |  | ||||||
|     } else if (source == QString("mem/physical/used")) { |  | ||||||
|         // used memory |  | ||||||
|         // mb |  | ||||||
|         m_map[source] = QString("memusedmb"); |  | ||||||
|         m_formater[QString("memusedmb")] = MemMBFormat; |  | ||||||
|         // gb |  | ||||||
|         m_map.insertMulti(source, QString("memusedgb")); |  | ||||||
|         m_formater[QString("memusedgb")] = MemGBFormat; |  | ||||||
|     } else if (source == QString("network/current/name")) { |  | ||||||
|         // network device |  | ||||||
|         m_map[source] = QString("netdev"); |  | ||||||
|         m_formater[QString("netdev")] = NoFormat; |  | ||||||
|     } else if (source.contains(netRegExp)) { |  | ||||||
|         // network speed |  | ||||||
|         QString type = source.contains(QString("receiver")) ? QString("down") |  | ||||||
|                                                             : QString("up"); |  | ||||||
|         int index |  | ||||||
|             = m_devices[QString("net")].indexOf(source.split(QChar('/'))[2]); |  | ||||||
|         if (index > -1) { |  | ||||||
|             // kb |  | ||||||
|             QString key = QString("%1kb%2").arg(type).arg(index); |  | ||||||
|             m_map[source] = key; |  | ||||||
|             m_formater[key] = Integer; |  | ||||||
|             // smart |  | ||||||
|             key = QString("%1%2").arg(type).arg(index); |  | ||||||
|             m_map.insertMulti(source, key); |  | ||||||
|             m_formater[key] = NetSmartFormat; |  | ||||||
|             // units |  | ||||||
|             key = QString("%1units%2").arg(type).arg(index); |  | ||||||
|             m_map.insertMulti(source, key); |  | ||||||
|             m_formater[key] = NetSmartUnits; |  | ||||||
|         } |  | ||||||
|     } else if (source.startsWith(QString("upgrade"))) { |  | ||||||
|         // package manager |  | ||||||
|         QString key = source; |  | ||||||
|         key.remove(QString("upgrade/")); |  | ||||||
|         m_map[source] = key; |  | ||||||
|         m_formater[key] = IntegerThree; |  | ||||||
|     } else if (source.startsWith(QString("player"))) { |  | ||||||
|         // player |  | ||||||
|         QString key = source; |  | ||||||
|         key.remove(QString("player/")); |  | ||||||
|         m_map[source] = key; |  | ||||||
|         m_formater[key] = NoFormat; |  | ||||||
|     } else if (source == QString("ps/running/count")) { |  | ||||||
|         // running processes count |  | ||||||
|         m_map[source] = QString("pscount"); |  | ||||||
|         m_formater[QString("pscount")] = NoFormat; |  | ||||||
|     } else if (source == QString("ps/running/list")) { |  | ||||||
|         // list of running processes |  | ||||||
|         m_map[source] = QString("ps"); |  | ||||||
|         m_formater[QString("ps")] = List; |  | ||||||
|     } else if (source == QString("ps/total/count")) { |  | ||||||
|         // total processes count |  | ||||||
|         m_map[source] = QString("pstotal"); |  | ||||||
|         m_formater[QString("pstotal")] = NoFormat; |  | ||||||
|     } else if (source.startsWith(QString("quotes"))) { |  | ||||||
|         // quotes |  | ||||||
|         QString key = source; |  | ||||||
|         key.remove(QString("quotes/")); |  | ||||||
|         m_map[source] = key; |  | ||||||
|         m_formater[key] = Quotes; |  | ||||||
|     } else if (source == QString("mem/swap/free")) { |  | ||||||
|         // free swap |  | ||||||
|         // mb |  | ||||||
|         m_map[source] = QString("swapfreemb"); |  | ||||||
|         m_formater[QString("swapfreemb")] = MemMBFormat; |  | ||||||
|         // gb |  | ||||||
|         m_map.insertMulti(source, QString("swapfreegb")); |  | ||||||
|         m_formater[QString("swapfreegb")] = MemGBFormat; |  | ||||||
|     } else if (source == QString("mem/swap/used")) { |  | ||||||
|         // used swap |  | ||||||
|         // mb |  | ||||||
|         m_map[source] = QString("swapmb"); |  | ||||||
|         m_formater[QString("swapmb")] = MemMBFormat; |  | ||||||
|         // gb |  | ||||||
|         m_map.insertMulti(source, QString("swapgb")); |  | ||||||
|         m_formater[QString("swapgb")] = MemGBFormat; |  | ||||||
|     } else if (source.startsWith(QString("lmsensors/"))) { |  | ||||||
|         // temperature |  | ||||||
|         int index = m_devices[QString("temp")].indexOf(source); |  | ||||||
|         // FIXME on DE initialization there are no units key |  | ||||||
|         if (units.isEmpty()) |  | ||||||
|             return QStringList() << QString("temp%1").arg(index); |  | ||||||
|         if (index > -1) { |  | ||||||
|             QString key = QString("temp%1").arg(index); |  | ||||||
|             m_map[source] = key; |  | ||||||
|             m_formater[key] = units == QString("°C") ? Temperature : Integer; |  | ||||||
|         } |  | ||||||
|     } else if (source == QString("Local")) { |  | ||||||
|         // time |  | ||||||
|         m_map[source] = QString("time"); |  | ||||||
|         m_formater[QString("time")] = Time; |  | ||||||
|         // custom time |  | ||||||
|         m_map.insertMulti(source, QString("ctime")); |  | ||||||
|         m_formater[QString("ctime")] = TimeCustom; |  | ||||||
|         // ISO time |  | ||||||
|         m_map.insertMulti(source, QString("isotime")); |  | ||||||
|         m_formater[QString("isotime")] = TimeISO; |  | ||||||
|         // long time |  | ||||||
|         m_map.insertMulti(source, QString("longtime")); |  | ||||||
|         m_formater[QString("longtime")] = TimeLong; |  | ||||||
|         // short time |  | ||||||
|         m_map.insertMulti(source, QString("shorttime")); |  | ||||||
|         m_formater[QString("shorttime")] = TimeShort; |  | ||||||
|     } else if (source == QString("system/uptime")) { |  | ||||||
|         // uptime |  | ||||||
|         m_map[source] = QString("uptime"); |  | ||||||
|         m_formater[QString("uptime")] = Uptime; |  | ||||||
|         // custom uptime |  | ||||||
|         m_map.insertMulti(source, QString("cuptime")); |  | ||||||
|         m_formater[QString("cuptime")] = UptimeCustom; |  | ||||||
|     } else if (source.startsWith(QString("weather/temperature"))) { |  | ||||||
|         // temperature |  | ||||||
|         QString key = source; |  | ||||||
|         key.remove(QString("weather/")); |  | ||||||
|         m_map[source] = key; |  | ||||||
|         m_formater[key] = Temperature; |  | ||||||
|     } else if (source.startsWith(QString("weather/"))) { |  | ||||||
|         // other weather |  | ||||||
|         QString key = source; |  | ||||||
|         key.remove(QString("weather/")); |  | ||||||
|         m_map[source] = key; |  | ||||||
|         m_formater[key] = NoFormat; |  | ||||||
|     } else if (source.startsWith(QString("load/load"))) { |  | ||||||
|         // load source |  | ||||||
|         QString key = source; |  | ||||||
|         key.remove(QString("load/")); |  | ||||||
|         m_map[source] = key; |  | ||||||
|         m_formater[key] = Temperature; |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     return keysFromSource(source); |  | ||||||
| } | } | ||||||
|  |  | ||||||
|  |  | ||||||
| @ -521,19 +247,19 @@ float AWKeysAggregator::temperature(const float temp) const | |||||||
|     qCDebug(LOG_AW) << "Temperature value" << temp; |     qCDebug(LOG_AW) << "Temperature value" << temp; | ||||||
|  |  | ||||||
|     float converted = temp; |     float converted = temp; | ||||||
|     if (m_tempUnits == QString("Celsius")) { |     if (m_tempUnits == "Celsius") { | ||||||
|     } else if (m_tempUnits == QString("Fahrenheit")) { |     } else if (m_tempUnits == "Fahrenheit") { | ||||||
|         converted = temp * 9.0 / 5.0 + 32.0; |         converted = temp * 9.0f / 5.0f + 32.0f; | ||||||
|     } else if (m_tempUnits == QString("Kelvin")) { |     } else if (m_tempUnits == "Kelvin") { | ||||||
|         converted = temp + 273.15; |         converted = temp + 273.15f; | ||||||
|     } else if (m_tempUnits == QString("Reaumur")) { |     } else if (m_tempUnits == "Reaumur") { | ||||||
|         converted = temp * 0.8; |         converted = temp * 0.8f; | ||||||
|     } else if (m_tempUnits == QString("cm^-1")) { |     } else if (m_tempUnits == "cm^-1") { | ||||||
|         converted = (temp + 273.15) * 0.695; |         converted = (temp + 273.15f) * 0.695f; | ||||||
|     } else if (m_tempUnits == QString("kJ/mol")) { |     } else if (m_tempUnits == "kJ/mol") { | ||||||
|         converted = (temp + 273.15) * 8.31; |         converted = (temp + 273.15f) * 8.31f; | ||||||
|     } else if (m_tempUnits == QString("kcal/mol")) { |     } else if (m_tempUnits == "kcal/mol") { | ||||||
|         converted = (temp + 273.15) * 1.98; |         converted = (temp + 273.15f) * 1.98f; | ||||||
|     } else { |     } else { | ||||||
|         qCWarning(LOG_AW) << "Invalid units" << m_tempUnits; |         qCWarning(LOG_AW) << "Invalid units" << m_tempUnits; | ||||||
|     } |     } | ||||||
|  | |||||||
| @ -22,8 +22,9 @@ | |||||||
| #include <QHash> | #include <QHash> | ||||||
| #include <QObject> | #include <QObject> | ||||||
|  |  | ||||||
| #include "version.h" |  | ||||||
|  |  | ||||||
|  | class AWFormatterHelper; | ||||||
|  | class AWDataEngineMapper; | ||||||
|  |  | ||||||
| class AWKeysAggregator : public QObject | class AWKeysAggregator : public QObject | ||||||
| { | { | ||||||
| @ -31,20 +32,22 @@ class AWKeysAggregator : public QObject | |||||||
|     Q_PROPERTY(QString acOffline MEMBER m_acOffline WRITE setAcOffline); |     Q_PROPERTY(QString acOffline MEMBER m_acOffline WRITE setAcOffline); | ||||||
|     Q_PROPERTY(QString acOnline MEMBER m_acOnline WRITE setAcOnline); |     Q_PROPERTY(QString acOnline MEMBER m_acOnline WRITE setAcOnline); | ||||||
|     Q_PROPERTY(QString customTime MEMBER m_customTime WRITE setCustomTime); |     Q_PROPERTY(QString customTime MEMBER m_customTime WRITE setCustomTime); | ||||||
|     Q_PROPERTY( |     Q_PROPERTY(QString customUptime MEMBER m_customUptime WRITE setCustomUptime); | ||||||
|         QString customUptime MEMBER m_customUptime WRITE setCustomUptime); |  | ||||||
|     Q_PROPERTY(QString tempUnits MEMBER m_tempUnits WRITE setTempUnits); |     Q_PROPERTY(QString tempUnits MEMBER m_tempUnits WRITE setTempUnits); | ||||||
|     Q_PROPERTY(bool translate MEMBER m_translate WRITE setTranslate); |     Q_PROPERTY(bool translate MEMBER m_translate WRITE setTranslate); | ||||||
|  |  | ||||||
|     enum FormaterType { | public: | ||||||
|         // general formaters |     enum class FormatterType { | ||||||
|         NoFormat = 0, |         // general formatters | ||||||
|  |         Custom, | ||||||
|  |         NoFormat, | ||||||
|         Float, |         Float, | ||||||
|         FloatTwoSymbols, |         FloatTwoSymbols, | ||||||
|         Integer, |         Integer, | ||||||
|  |         IntegerFive, | ||||||
|         IntegerThree, |         IntegerThree, | ||||||
|         List, |         List, | ||||||
|         // unit specific formaters |         // unit specific formatters | ||||||
|         ACFormat, |         ACFormat, | ||||||
|         MemGBFormat, |         MemGBFormat, | ||||||
|         MemMBFormat, |         MemMBFormat, | ||||||
| @ -57,39 +60,39 @@ class AWKeysAggregator : public QObject | |||||||
|         TimeISO, |         TimeISO, | ||||||
|         TimeLong, |         TimeLong, | ||||||
|         TimeShort, |         TimeShort, | ||||||
|  |         Timestamp, | ||||||
|         Uptime, |         Uptime, | ||||||
|         UptimeCustom |         UptimeCustom | ||||||
|     }; |     }; | ||||||
|  |  | ||||||
| public: |     explicit AWKeysAggregator(QObject *_parent = nullptr); | ||||||
|     explicit AWKeysAggregator(QObject *parent = nullptr); |     ~AWKeysAggregator() override; | ||||||
|     virtual ~AWKeysAggregator(); |     void initFormatters(); | ||||||
|     // get methods |     // get methods | ||||||
|     QString formater(const QVariant &data, const QString &key) const; |     [[nodiscard]] QString formatter(const QVariant &_data, const QString &_key, bool replaceSpace) const; | ||||||
|     QStringList keysFromSource(const QString &source) const; |     [[nodiscard]] QStringList keysFromSource(const QString &_source) const; | ||||||
|     // set methods |     // set methods | ||||||
|     void setAcOffline(const QString inactive); |     void setAcOffline(const QString &_inactive); | ||||||
|     void setAcOnline(const QString active); |     void setAcOnline(const QString &_active); | ||||||
|     void setCustomTime(const QString customTime); |     void setCustomTime(const QString &_customTime); | ||||||
|     void setCustomUptime(const QString customUptime); |     void setCustomUptime(const QString &_customUptime); | ||||||
|     void setDevices(const QHash<QString, QStringList> devices); |     void setDevices(const QHash<QString, QStringList> &_devices); | ||||||
|     void setTempUnits(const QString units); |     void setTempUnits(const QString &_units); | ||||||
|     void setTranslate(const bool translate); |     void setTranslate(bool _translate); | ||||||
|  |  | ||||||
| public slots: | public slots: | ||||||
|     QStringList registerSource(const QString &source, const QString &units); |     QStringList registerSource(const QString &_source, const QString &_units, const QStringList &_keys); | ||||||
|  |  | ||||||
| private: | private: | ||||||
|     float temperature(const float temp) const; |     [[nodiscard]] float temperature(float temp) const; | ||||||
|     QStringList timeKeys = QString(TIME_KEYS).split(QChar(',')); |     AWFormatterHelper *m_customFormatters = nullptr; | ||||||
|  |     AWDataEngineMapper *m_mapper = nullptr; | ||||||
|  |     QStringList m_timeKeys; | ||||||
|     // variables |     // variables | ||||||
|     QString m_acOffline; |     QString m_acOffline; | ||||||
|     QString m_acOnline; |     QString m_acOnline; | ||||||
|     QString m_customTime; |     QString m_customTime; | ||||||
|     QString m_customUptime; |     QString m_customUptime; | ||||||
|     QHash<QString, QStringList> m_devices; |  | ||||||
|     QHash<QString, FormaterType> m_formater; |  | ||||||
|     QHash<QString, QString> m_map; |  | ||||||
|     QString m_tempUnits; |     QString m_tempUnits; | ||||||
|     bool m_translate = false; |     bool m_translate = false; | ||||||
| }; | }; | ||||||
|  | |||||||
							
								
								
									
										51
									
								
								sources/awesome-widget/plugin/awpairconfigfactory.cpp
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										51
									
								
								sources/awesome-widget/plugin/awpairconfigfactory.cpp
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,51 @@ | |||||||
|  | /*************************************************************************** | ||||||
|  |  *   This file is part of awesome-widgets                                  * | ||||||
|  |  *                                                                         * | ||||||
|  |  *   awesome-widgets is free software: you can redistribute it and/or      * | ||||||
|  |  *   modify it under the terms of the GNU General Public License as        * | ||||||
|  |  *   published by the Free Software Foundation, either version 3 of the    * | ||||||
|  |  *   License, or (at your option) any later version.                       * | ||||||
|  |  *                                                                         * | ||||||
|  |  *   awesome-widgets is distributed in the hope that it will be useful,    * | ||||||
|  |  *   but WITHOUT ANY WARRANTY; without even the implied warranty of        * | ||||||
|  |  *   MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the         * | ||||||
|  |  *   GNU General Public License for more details.                          * | ||||||
|  |  *                                                                         * | ||||||
|  |  *   You should have received a copy of the GNU General Public License     * | ||||||
|  |  *   along with awesome-widgets. If not, see http://www.gnu.org/licenses/  * | ||||||
|  |  ***************************************************************************/ | ||||||
|  |  | ||||||
|  | #include "awpairconfigfactory.h" | ||||||
|  |  | ||||||
|  | #include "awcustomkeysconfig.h" | ||||||
|  | #include "awdebug.h" | ||||||
|  | #include "awformatterconfig.h" | ||||||
|  |  | ||||||
|  |  | ||||||
|  | AWPairConfigFactory::AWPairConfigFactory(QObject *_parent) | ||||||
|  |     : QObject(_parent) | ||||||
|  | { | ||||||
|  |     qCDebug(LOG_AW) << __PRETTY_FUNCTION__; | ||||||
|  | } | ||||||
|  |  | ||||||
|  |  | ||||||
|  | AWPairConfigFactory::~AWPairConfigFactory() | ||||||
|  | { | ||||||
|  |     qCDebug(LOG_AW) << __PRETTY_FUNCTION__; | ||||||
|  | } | ||||||
|  |  | ||||||
|  |  | ||||||
|  | void AWPairConfigFactory::showFormatterDialog(const QStringList &_keys) | ||||||
|  | { | ||||||
|  |     auto *config = new AWFormatterConfig(nullptr, _keys); | ||||||
|  |     config->showDialog(); | ||||||
|  |     config->deleteLater(); | ||||||
|  | } | ||||||
|  |  | ||||||
|  |  | ||||||
|  | void AWPairConfigFactory::showKeysDialog(const QStringList &_keys) | ||||||
|  | { | ||||||
|  |     auto *config = new AWCustomKeysConfig(nullptr, _keys); | ||||||
|  |     config->showDialog(); | ||||||
|  |     config->deleteLater(); | ||||||
|  | } | ||||||
| @ -15,24 +15,25 @@ | |||||||
|  *   along with awesome-widgets. If not, see http://www.gnu.org/licenses/  *
 |  *   along with awesome-widgets. If not, see http://www.gnu.org/licenses/  *
 | ||||||
|  ***************************************************************************/ |  ***************************************************************************/ | ||||||
| 
 | 
 | ||||||
| #ifndef UPDATESOURCE_H | 
 | ||||||
| #define UPDATESOURCE_H | #ifndef AWPAIRCONFIGFACTORY_H | ||||||
|  | #define AWPAIRCONFIGFACTORY_H | ||||||
| 
 | 
 | ||||||
| #include <QObject> | #include <QObject> | ||||||
| 
 | 
 | ||||||
| #include "abstractextsysmonsource.h" |  | ||||||
| 
 | 
 | ||||||
| 
 | class AWPairConfigFactory : public QObject | ||||||
| class UpdateSource : public AbstractExtSysMonSource |  | ||||||
| { | { | ||||||
|  |     Q_OBJECT | ||||||
|  | 
 | ||||||
| public: | public: | ||||||
|     explicit UpdateSource(QObject *parent, const QStringList args); |     explicit AWPairConfigFactory(QObject *_parent = nullptr); | ||||||
|     virtual ~UpdateSource(); |     ~AWPairConfigFactory() override; | ||||||
|     QVariant data(QString source); |     Q_INVOKABLE static void showFormatterDialog(const QStringList &_keys); | ||||||
|     QVariantMap initialData(QString source) const; |     Q_INVOKABLE static void showKeysDialog(const QStringList &_keys); | ||||||
|     void run(){}; | 
 | ||||||
|     QStringList sources() const; | private: | ||||||
| }; | }; | ||||||
| 
 | 
 | ||||||
| 
 | 
 | ||||||
| #endif /* UPDATESOURCE_H */ | #endif /* AWPAIRCONFIGFACTORY_H */ | ||||||
							
								
								
									
										270
									
								
								sources/awesome-widget/plugin/awpatternfunctions.cpp
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										270
									
								
								sources/awesome-widget/plugin/awpatternfunctions.cpp
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,270 @@ | |||||||
|  | /*************************************************************************** | ||||||
|  |  *   This file is part of awesome-widgets                                  * | ||||||
|  |  *                                                                         * | ||||||
|  |  *   awesome-widgets is free software: you can redistribute it and/or      * | ||||||
|  |  *   modify it under the terms of the GNU General Public License as        * | ||||||
|  |  *   published by the Free Software Foundation, either version 3 of the    * | ||||||
|  |  *   License, or (at your option) any later version.                       * | ||||||
|  |  *                                                                         * | ||||||
|  |  *   awesome-widgets is distributed in the hope that it will be useful,    * | ||||||
|  |  *   but WITHOUT ANY WARRANTY; without even the implied warranty of        * | ||||||
|  |  *   MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the         * | ||||||
|  |  *   GNU General Public License for more details.                          * | ||||||
|  |  *                                                                         * | ||||||
|  |  *   You should have received a copy of the GNU General Public License     * | ||||||
|  |  *   along with awesome-widgets. If not, see http://www.gnu.org/licenses/  * | ||||||
|  |  ***************************************************************************/ | ||||||
|  |  | ||||||
|  | #include "awpatternfunctions.h" | ||||||
|  |  | ||||||
|  | #include <QJSEngine> | ||||||
|  | #include <QRegularExpression> | ||||||
|  |  | ||||||
|  | #include "awdebug.h" | ||||||
|  | #include "awkeysaggregator.h" | ||||||
|  |  | ||||||
|  |  | ||||||
|  | QString AWPatternFunctions::expandLambdas(QString _code, AWKeysAggregator *_aggregator, const QVariantHash &_metadata, | ||||||
|  |                                           const QStringList &_usedKeys) | ||||||
|  | { | ||||||
|  |     qCDebug(LOG_AW) << "Expand lambdas in" << _code; | ||||||
|  |  | ||||||
|  |     QJSEngine engine; | ||||||
|  |     // apply $this values | ||||||
|  |     _code.replace("$this", _metadata[_code].toString()); | ||||||
|  |     // parsed values | ||||||
|  |     for (auto &lambdaKey : _usedKeys) | ||||||
|  |         _code.replace(QString("$%1").arg(lambdaKey), _aggregator->formatter(_metadata[lambdaKey], lambdaKey, false)); | ||||||
|  |     qCInfo(LOG_AW) << "Expression" << _code; | ||||||
|  |     QJSValue result = engine.evaluate(_code); | ||||||
|  |     if (result.isError()) { | ||||||
|  |         qCWarning(LOG_AW) << "Uncaught exception at line" << result.property("lineNumber").toInt() << ":" | ||||||
|  |                           << result.toString(); | ||||||
|  |         return ""; | ||||||
|  |     } else { | ||||||
|  |         return result.toString(); | ||||||
|  |     } | ||||||
|  | } | ||||||
|  |  | ||||||
|  |  | ||||||
|  | QString AWPatternFunctions::expandTemplates(QString _code) | ||||||
|  | { | ||||||
|  |     qCDebug(LOG_AW) << "Expand templates in" << _code; | ||||||
|  |  | ||||||
|  |     // match the following construction $template{{some code here}} | ||||||
|  |     QRegularExpression templatesRegexp(R"(\$template\{\{(?<body>.*?)\}\})"); | ||||||
|  |     templatesRegexp.setPatternOptions(QRegularExpression::DotMatchesEverythingOption); | ||||||
|  |  | ||||||
|  |     QRegularExpressionMatchIterator it = templatesRegexp.globalMatch(_code); | ||||||
|  |     while (it.hasNext()) { | ||||||
|  |         QRegularExpressionMatch match = it.next(); | ||||||
|  |         QString body = match.captured("body"); | ||||||
|  |  | ||||||
|  |         QJSEngine engine; | ||||||
|  |         qCInfo(LOG_AW) << "Expression" << body; | ||||||
|  |         QJSValue result = engine.evaluate(body); | ||||||
|  |         QString templateResult = ""; | ||||||
|  |         if (result.isError()) { | ||||||
|  |             qCWarning(LOG_AW) << "Uncaught exception at line" << result.property("lineNumber").toInt() << ":" | ||||||
|  |                               << result.toString(); | ||||||
|  |         } else { | ||||||
|  |             templateResult = result.toString(); | ||||||
|  |         } | ||||||
|  |  | ||||||
|  |         // replace template | ||||||
|  |         _code.replace(match.captured(), templateResult); | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     return _code; | ||||||
|  | } | ||||||
|  |  | ||||||
|  |  | ||||||
|  | QList<AWPatternFunctions::AWFunction> AWPatternFunctions::findFunctionCalls(const QString &_function, | ||||||
|  |                                                                             const QString &_code) | ||||||
|  | { | ||||||
|  |     qCDebug(LOG_AW) << "Looking for function" << _function << "in" << _code; | ||||||
|  |  | ||||||
|  |     // I suggest the following regex for the internal functions | ||||||
|  |     // $aw_function_name<some args here if any>{{function body}} | ||||||
|  |     // * args should be always comma separated (e.g. commas are not supported | ||||||
|  |     // in this field if they are not screened by $, i.e. '$,' | ||||||
|  |     // * body depends on the function name, double brackets should be screened | ||||||
|  |     // by using $, e.g. ${ | ||||||
|  |     QRegularExpression regex(QString(R"(\$%1\<(?<args>.*?)\>\{\{(?<body>.*?)\}\})").arg(_function)); | ||||||
|  |     regex.setPatternOptions(QRegularExpression::DotMatchesEverythingOption); | ||||||
|  |  | ||||||
|  |     QList<AWPatternFunctions::AWFunction> foundFunctions; | ||||||
|  |     QRegularExpressionMatchIterator it = regex.globalMatch(_code); | ||||||
|  |     while (it.hasNext()) { | ||||||
|  |         QRegularExpressionMatch match = it.next(); | ||||||
|  |  | ||||||
|  |         AWPatternFunctions::AWFunction metadata; | ||||||
|  |         // work with args | ||||||
|  |         QString argsString = match.captured("args"); | ||||||
|  |         if (argsString.isEmpty()) { | ||||||
|  |             metadata.args = QStringList(); | ||||||
|  |         } else { | ||||||
|  |             // replace '$,' to 0x1d | ||||||
|  |             argsString.replace("$,", QString(0x1d)); | ||||||
|  |             QStringList args = argsString.split(','); | ||||||
|  |             std::for_each(args.begin(), args.end(), [](QString &arg) { arg.replace(QString(0x1d), ","); }); | ||||||
|  |             metadata.args = args; | ||||||
|  |         } | ||||||
|  |         // other variables | ||||||
|  |         metadata.body = match.captured("body"); | ||||||
|  |         metadata.what = match.captured(); | ||||||
|  |         // replace brackets | ||||||
|  |         metadata.body.replace("${", "{"); | ||||||
|  |         metadata.body.replace("$}", "}"); | ||||||
|  |         foundFunctions.append(metadata); | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     return foundFunctions; | ||||||
|  | } | ||||||
|  |  | ||||||
|  |  | ||||||
|  | QString AWPatternFunctions::insertAllKeys(QString _code, const QStringList &_keys) | ||||||
|  | { | ||||||
|  |     qCDebug(LOG_AW) << "Looking for keys in code" << _code << "using list" << _keys; | ||||||
|  |  | ||||||
|  |     QList<AWPatternFunctions::AWFunction> found = AWPatternFunctions::findFunctionCalls("aw_all", _code); | ||||||
|  |     for (auto &function : found) { | ||||||
|  |         QString separator = function.args.isEmpty() ? "," : function.args.at(0); | ||||||
|  |         QStringList required = _keys.filter(QRegExp(function.body)); | ||||||
|  |         std::for_each(required.begin(), required.end(), [](QString &value) { value = QString("%1: $%1").arg(value); }); | ||||||
|  |  | ||||||
|  |         _code.replace(function.what, required.join(separator)); | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     return _code; | ||||||
|  | } | ||||||
|  |  | ||||||
|  |  | ||||||
|  | QString AWPatternFunctions::insertKeyCount(QString _code, const QStringList &_keys) | ||||||
|  | { | ||||||
|  |     qCDebug(LOG_AW) << "Looking for count in code" << _code << "using list" << _keys; | ||||||
|  |  | ||||||
|  |     QList<AWPatternFunctions::AWFunction> found = AWPatternFunctions::findFunctionCalls("aw_count", _code); | ||||||
|  |     for (auto &function : found) { | ||||||
|  |         int count = _keys.filter(QRegExp(function.body)).count(); | ||||||
|  |  | ||||||
|  |         _code.replace(function.what, QString::number(count)); | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     return _code; | ||||||
|  | } | ||||||
|  |  | ||||||
|  |  | ||||||
|  | QString AWPatternFunctions::insertKeyNames(QString _code, const QStringList &_keys) | ||||||
|  | { | ||||||
|  |     qCDebug(LOG_AW) << "Looking for key names in code" << _code << "using list" << _keys; | ||||||
|  |  | ||||||
|  |     QList<AWPatternFunctions::AWFunction> found = AWPatternFunctions::findFunctionCalls("aw_names", _code); | ||||||
|  |     for (auto &function : found) { | ||||||
|  |         QString separator = function.args.isEmpty() ? "," : function.args.at(0); | ||||||
|  |         QStringList required = _keys.filter(QRegExp(function.body)); | ||||||
|  |  | ||||||
|  |         _code.replace(function.what, required.join(separator)); | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     return _code; | ||||||
|  | } | ||||||
|  |  | ||||||
|  |  | ||||||
|  | QString AWPatternFunctions::insertKeys(QString _code, const QStringList &_keys) | ||||||
|  | { | ||||||
|  |     qCDebug(LOG_AW) << "Looking for keys in code" << _code << "using list" << _keys; | ||||||
|  |  | ||||||
|  |     QList<AWPatternFunctions::AWFunction> found = AWPatternFunctions::findFunctionCalls("aw_keys", _code); | ||||||
|  |     for (auto &function : found) { | ||||||
|  |         QString separator = function.args.isEmpty() ? "," : function.args.at(0); | ||||||
|  |         QStringList required = _keys.filter(QRegExp(function.body)); | ||||||
|  |         std::for_each(required.begin(), required.end(), [](QString &value) { value = QString("$%1").arg(value); }); | ||||||
|  |  | ||||||
|  |         _code.replace(function.what, required.join(separator)); | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     return _code; | ||||||
|  | } | ||||||
|  |  | ||||||
|  |  | ||||||
|  | QString AWPatternFunctions::insertMacros(QString _code) | ||||||
|  | { | ||||||
|  |     qCDebug(LOG_AW) << "Looking for macros in code" << _code; | ||||||
|  |  | ||||||
|  |     QList<AWPatternFunctions::AWFunction> found = AWPatternFunctions::findFunctionCalls("aw_macro", _code); | ||||||
|  |     for (auto ¯o : found) { | ||||||
|  |         // get macro params | ||||||
|  |         if (macro.args.isEmpty()) { | ||||||
|  |             qCWarning(LOG_AW) << "No macro name found for" << macro.what; | ||||||
|  |             continue; | ||||||
|  |         } | ||||||
|  |         QString name = macro.args.takeFirst(); | ||||||
|  |         // find macro usage | ||||||
|  |         QList<AWPatternFunctions::AWFunction> macroUsage | ||||||
|  |             = AWPatternFunctions::findFunctionCalls(QString("aw_macro_%1").arg(name), _code); | ||||||
|  |         for (auto &function : macroUsage) { | ||||||
|  |             if (function.args.count() != macro.args.count()) { | ||||||
|  |                 qCWarning(LOG_AW) << "Invalid args count found for call" << function.what << "with macro" << macro.what; | ||||||
|  |                 continue; | ||||||
|  |             } | ||||||
|  |             // generate body to replace | ||||||
|  |             QString result = macro.body; | ||||||
|  |             std::for_each(macro.args.cbegin(), macro.args.cend(), [&result, macro, function](const QString &arg) { | ||||||
|  |                 int index = macro.args.indexOf(arg); | ||||||
|  |                 result.replace(QString("$%1").arg(arg), function.args.at(index)); | ||||||
|  |             }); | ||||||
|  |             // do replace | ||||||
|  |             _code.replace(function.what, result); | ||||||
|  |         } | ||||||
|  |  | ||||||
|  |         // remove macro from source pattern | ||||||
|  |         _code.remove(macro.what); | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     return _code; | ||||||
|  | } | ||||||
|  |  | ||||||
|  |  | ||||||
|  | QStringList AWPatternFunctions::findKeys(const QString &_code, const QStringList &_keys, const bool _isBars) | ||||||
|  | { | ||||||
|  |     qCDebug(LOG_AW) << "Looking for keys in code" << _code << "using list" << _keys; | ||||||
|  |  | ||||||
|  |     QStringList selectedKeys; | ||||||
|  |     QString replacedCode = _code; | ||||||
|  |     for (auto &key : _keys) | ||||||
|  |         if ((key.startsWith("bar") == _isBars) && (replacedCode.contains(QString("$%1").arg(key)))) { | ||||||
|  |             qCInfo(LOG_AW) << "Found key" << key << "with bar enabled" << _isBars; | ||||||
|  |             selectedKeys.append(key); | ||||||
|  |             replacedCode.replace(QString("$%1").arg(key), ""); | ||||||
|  |         } | ||||||
|  |     if (selectedKeys.isEmpty()) | ||||||
|  |         qCWarning(LOG_AW) << "No keys found"; | ||||||
|  |  | ||||||
|  |     return selectedKeys; | ||||||
|  | } | ||||||
|  |  | ||||||
|  |  | ||||||
|  | QStringList AWPatternFunctions::findLambdas(const QString &_code) | ||||||
|  | { | ||||||
|  |     qCDebug(LOG_AW) << "Looking for lambdas in code" << _code; | ||||||
|  |  | ||||||
|  |     QStringList selectedKeys; | ||||||
|  |     // match the following construction ${{some code here}} | ||||||
|  |     QRegularExpression lambdaRegexp(R"(\$\{\{(?<body>.*?)\}\})"); | ||||||
|  |     lambdaRegexp.setPatternOptions(QRegularExpression::DotMatchesEverythingOption); | ||||||
|  |  | ||||||
|  |     QRegularExpressionMatchIterator it = lambdaRegexp.globalMatch(_code); | ||||||
|  |     while (it.hasNext()) { | ||||||
|  |         QRegularExpressionMatch match = it.next(); | ||||||
|  |         QString lambda = match.captured("body"); | ||||||
|  |  | ||||||
|  |         // append | ||||||
|  |         qCInfo(LOG_AW) << "Found lambda" << lambda; | ||||||
|  |         selectedKeys.append(lambda); | ||||||
|  |     } | ||||||
|  |     if (selectedKeys.isEmpty()) | ||||||
|  |         qCWarning(LOG_AW) << "No lambdas found"; | ||||||
|  |  | ||||||
|  |     return selectedKeys; | ||||||
|  | } | ||||||
							
								
								
									
										52
									
								
								sources/awesome-widget/plugin/awpatternfunctions.h
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										52
									
								
								sources/awesome-widget/plugin/awpatternfunctions.h
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,52 @@ | |||||||
|  | /*************************************************************************** | ||||||
|  |  *   This file is part of awesome-widgets                                  * | ||||||
|  |  *                                                                         * | ||||||
|  |  *   awesome-widgets is free software: you can redistribute it and/or      * | ||||||
|  |  *   modify it under the terms of the GNU General Public License as        * | ||||||
|  |  *   published by the Free Software Foundation, either version 3 of the    * | ||||||
|  |  *   License, or (at your option) any later version.                       * | ||||||
|  |  *                                                                         * | ||||||
|  |  *   awesome-widgets is distributed in the hope that it will be useful,    * | ||||||
|  |  *   but WITHOUT ANY WARRANTY; without even the implied warranty of        * | ||||||
|  |  *   MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the         * | ||||||
|  |  *   GNU General Public License for more details.                          * | ||||||
|  |  *                                                                         * | ||||||
|  |  *   You should have received a copy of the GNU General Public License     * | ||||||
|  |  *   along with awesome-widgets. If not, see http://www.gnu.org/licenses/  * | ||||||
|  |  ***************************************************************************/ | ||||||
|  |  | ||||||
|  |  | ||||||
|  | #ifndef AWPATTERNFUNCTIONS_H | ||||||
|  | #define AWPATTERNFUNCTIONS_H | ||||||
|  |  | ||||||
|  | #include <QString> | ||||||
|  | #include <QVariant> | ||||||
|  |  | ||||||
|  |  | ||||||
|  | class AWKeysAggregator; | ||||||
|  |  | ||||||
|  | namespace AWPatternFunctions | ||||||
|  | { | ||||||
|  | typedef struct { | ||||||
|  |     QStringList args; | ||||||
|  |     QString body; | ||||||
|  |     QString what; | ||||||
|  | } AWFunction; | ||||||
|  |  | ||||||
|  | // insert methods | ||||||
|  | QString expandLambdas(QString _code, AWKeysAggregator *_aggregator, const QVariantHash &_metadata, | ||||||
|  |                       const QStringList &_usedKeys); | ||||||
|  | QString expandTemplates(QString _code); | ||||||
|  | QList<AWFunction> findFunctionCalls(const QString &_function, const QString &_code); | ||||||
|  | QString insertAllKeys(QString _code, const QStringList &_keys); | ||||||
|  | QString insertKeyCount(QString _code, const QStringList &_keys); | ||||||
|  | QString insertKeyNames(QString _code, const QStringList &_keys); | ||||||
|  | QString insertKeys(QString _code, const QStringList &_keys); | ||||||
|  | QString insertMacros(QString _code); | ||||||
|  | // find methods | ||||||
|  | QStringList findKeys(const QString &_code, const QStringList &_keys, bool _isBars); | ||||||
|  | QStringList findLambdas(const QString &_code); | ||||||
|  | } // namespace AWPatternFunctions | ||||||
|  |  | ||||||
|  |  | ||||||
|  | #endif /* AWPATTERNFUNCTIONS_H */ | ||||||
							
								
								
									
										179
									
								
								sources/awesome-widget/plugin/awtelemetryhandler.cpp
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										179
									
								
								sources/awesome-widget/plugin/awtelemetryhandler.cpp
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,179 @@ | |||||||
|  | /*************************************************************************** | ||||||
|  |  *   This file is part of awesome-widgets                                  * | ||||||
|  |  *                                                                         * | ||||||
|  |  *   awesome-widgets is free software: you can redistribute it and/or      * | ||||||
|  |  *   modify it under the terms of the GNU General Public License as        * | ||||||
|  |  *   published by the Free Software Foundation, either version 3 of the    * | ||||||
|  |  *   License, or (at your option) any later version.                       * | ||||||
|  |  *                                                                         * | ||||||
|  |  *   awesome-widgets is distributed in the hope that it will be useful,    * | ||||||
|  |  *   but WITHOUT ANY WARRANTY; without even the implied warranty of        * | ||||||
|  |  *   MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the         * | ||||||
|  |  *   GNU General Public License for more details.                          * | ||||||
|  |  *                                                                         * | ||||||
|  |  *   You should have received a copy of the GNU General Public License     * | ||||||
|  |  *   along with awesome-widgets. If not, see http://www.gnu.org/licenses/  * | ||||||
|  |  ***************************************************************************/ | ||||||
|  |  | ||||||
|  | #include "awtelemetryhandler.h" | ||||||
|  |  | ||||||
|  | #include <QJsonDocument> | ||||||
|  | #include <QNetworkAccessManager> | ||||||
|  | #include <QNetworkReply> | ||||||
|  | #include <QSettings> | ||||||
|  | #include <QStandardPaths> | ||||||
|  | #include <QUuid> | ||||||
|  |  | ||||||
|  | #include "awdebug.h" | ||||||
|  |  | ||||||
|  |  | ||||||
|  | AWTelemetryHandler::AWTelemetryHandler(QObject *_parent, const QString &_clientId) | ||||||
|  |     : QObject(_parent) | ||||||
|  | { | ||||||
|  |     qCDebug(LOG_AW) << __PRETTY_FUNCTION__; | ||||||
|  |  | ||||||
|  |     m_localFile = QString("%1/awesomewidgets/telemetry.ini") | ||||||
|  |                       .arg(QStandardPaths::writableLocation(QStandardPaths::GenericDataLocation)); | ||||||
|  |  | ||||||
|  |     // override client id if any | ||||||
|  |     if (!_clientId.isEmpty()) | ||||||
|  |         m_clientId = _clientId; | ||||||
|  | } | ||||||
|  |  | ||||||
|  |  | ||||||
|  | AWTelemetryHandler::~AWTelemetryHandler() | ||||||
|  | { | ||||||
|  |     qCDebug(LOG_AW) << __PRETTY_FUNCTION__; | ||||||
|  | } | ||||||
|  |  | ||||||
|  |  | ||||||
|  | QStringList AWTelemetryHandler::get(const QString &_group) const | ||||||
|  | { | ||||||
|  |     qCDebug(LOG_AW) << "Get stored data for group" << _group; | ||||||
|  |  | ||||||
|  |     QStringList values; | ||||||
|  |  | ||||||
|  |     QSettings settings(m_localFile, QSettings::IniFormat); | ||||||
|  |     settings.beginGroup(_group); | ||||||
|  |     for (auto &key : settings.childKeys()) | ||||||
|  |         values.append(settings.value(key).toString()); | ||||||
|  |     settings.endGroup(); | ||||||
|  |  | ||||||
|  |     return values; | ||||||
|  | } | ||||||
|  |  | ||||||
|  |  | ||||||
|  | QString AWTelemetryHandler::getLast(const QString &_group) const | ||||||
|  | { | ||||||
|  |     qCDebug(LOG_AW) << "Get last stored data for group" << _group; | ||||||
|  |  | ||||||
|  |     return get(_group).last(); | ||||||
|  | } | ||||||
|  |  | ||||||
|  |  | ||||||
|  | void AWTelemetryHandler::init(const int _count, const bool _enableRemote, const QString &_clientId) | ||||||
|  | { | ||||||
|  |     qCDebug(LOG_AW) << "Init telemetry with count" << _count << "enable remote" << _enableRemote << "client ID" | ||||||
|  |                     << _clientId; | ||||||
|  |  | ||||||
|  |     m_storeCount = _count; | ||||||
|  |     m_uploadEnabled = _enableRemote; | ||||||
|  |     m_clientId = _clientId; | ||||||
|  | } | ||||||
|  |  | ||||||
|  |  | ||||||
|  | bool AWTelemetryHandler::put(const QString &_group, const QString &_value) const | ||||||
|  | { | ||||||
|  |     qCDebug(LOG_AW) << "Store data with group" << _group << "and value" << _value; | ||||||
|  |  | ||||||
|  |     QSettings settings(m_localFile, QSettings::IniFormat); | ||||||
|  |     settings.beginGroup(_group); | ||||||
|  |     // values will be stored as num=value inside specified group | ||||||
|  |     // load all values to memory | ||||||
|  |     QStringList saved; | ||||||
|  |     for (auto &key : settings.childKeys()) | ||||||
|  |         saved.append(settings.value(key).toString()); | ||||||
|  |     // check if this value is already saved | ||||||
|  |     if (saved.contains(_value)) { | ||||||
|  |         qCInfo(LOG_AW) << "Configuration" << _value << "for group" << _group << "is already saved"; | ||||||
|  |         return false; | ||||||
|  |     } | ||||||
|  |     saved.append(_value); | ||||||
|  |     // remove old ones | ||||||
|  |     while (saved.count() > m_storeCount) | ||||||
|  |         saved.takeFirst(); | ||||||
|  |     // clear group | ||||||
|  |     settings.remove(""); | ||||||
|  |     // and save now | ||||||
|  |     for (auto &val : saved) { | ||||||
|  |         QString key = getKey(settings.childKeys().count()); | ||||||
|  |         settings.setValue(key, val); | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     // sync settings | ||||||
|  |     settings.endGroup(); | ||||||
|  |     settings.sync(); | ||||||
|  |     // return status | ||||||
|  |     return (settings.status() == QSettings::NoError); | ||||||
|  | } | ||||||
|  |  | ||||||
|  |  | ||||||
|  | void AWTelemetryHandler::uploadTelemetry(const QString &_group, const QString &_value) | ||||||
|  | { | ||||||
|  |     qCDebug(LOG_AW) << "Upload data with group" << _group << "and value" << _value; | ||||||
|  |     if (!m_uploadEnabled) { | ||||||
|  |         qCInfo(LOG_AW) << "Upload disabled by configuration"; | ||||||
|  |         return; | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     auto *manager = new QNetworkAccessManager(nullptr); | ||||||
|  |     connect(manager, SIGNAL(finished(QNetworkReply *)), this, SLOT(telemetryReplyRecieved(QNetworkReply *))); | ||||||
|  |  | ||||||
|  |     QUrl url(REMOTE_TELEMETRY_URL); | ||||||
|  |     QNetworkRequest request(url); | ||||||
|  |     request.setHeader(QNetworkRequest::ContentTypeHeader, "application/json"); | ||||||
|  |  | ||||||
|  |     // generate payload | ||||||
|  |     QVariantMap payload; | ||||||
|  |     payload["api"] = AW_TELEMETRY_API; | ||||||
|  |     payload["client_id"] = m_clientId; | ||||||
|  |     payload["metadata"] = _value; | ||||||
|  |     payload["type"] = _group; | ||||||
|  |     // convert to QByteArray to send request | ||||||
|  |     QByteArray data = QJsonDocument::fromVariant(payload).toJson(QJsonDocument::Compact); | ||||||
|  |     qCInfo(LOG_AW) << "Send request with body" << data.data() << "and size" << data.size(); | ||||||
|  |  | ||||||
|  |     manager->post(request, data); | ||||||
|  | } | ||||||
|  |  | ||||||
|  |  | ||||||
|  | void AWTelemetryHandler::telemetryReplyRecieved(QNetworkReply *_reply) | ||||||
|  | { | ||||||
|  |     if (_reply->error() != QNetworkReply::NoError) { | ||||||
|  |         qCWarning(LOG_AW) << "An error occurs" << _reply->error() << "with message" << _reply->errorString(); | ||||||
|  |         return; | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     QJsonParseError error{}; | ||||||
|  |     QJsonDocument jsonDoc = QJsonDocument::fromJson(_reply->readAll(), &error); | ||||||
|  |     if (error.error != QJsonParseError::NoError) { | ||||||
|  |         qCWarning(LOG_AW) << "Parse error" << error.errorString(); | ||||||
|  |         return; | ||||||
|  |     } | ||||||
|  |     _reply->deleteLater(); | ||||||
|  |  | ||||||
|  |     // convert to map | ||||||
|  |     QVariantMap response = jsonDoc.toVariant().toMap(); | ||||||
|  |     QString message = response["message"].toString(); | ||||||
|  |     qCInfo(LOG_AW) << "Server reply on telemetry" << message; | ||||||
|  |  | ||||||
|  |     return emit(replyReceived(message)); | ||||||
|  | } | ||||||
|  |  | ||||||
|  |  | ||||||
|  | QString AWTelemetryHandler::getKey(const int _count) | ||||||
|  | { | ||||||
|  |     qCDebug(LOG_AW) << "Get key for keys count" << _count; | ||||||
|  |  | ||||||
|  |     return QString("%1").arg(_count, 3, 10, QChar('0')); | ||||||
|  | } | ||||||
							
								
								
									
										57
									
								
								sources/awesome-widget/plugin/awtelemetryhandler.h
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										57
									
								
								sources/awesome-widget/plugin/awtelemetryhandler.h
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,57 @@ | |||||||
|  | /*************************************************************************** | ||||||
|  |  *   This file is part of awesome-widgets                                  * | ||||||
|  |  *                                                                         * | ||||||
|  |  *   awesome-widgets is free software: you can redistribute it and/or      * | ||||||
|  |  *   modify it under the terms of the GNU General Public License as        * | ||||||
|  |  *   published by the Free Software Foundation, either version 3 of the    * | ||||||
|  |  *   License, or (at your option) any later version.                       * | ||||||
|  |  *                                                                         * | ||||||
|  |  *   awesome-widgets is distributed in the hope that it will be useful,    * | ||||||
|  |  *   but WITHOUT ANY WARRANTY; without even the implied warranty of        * | ||||||
|  |  *   MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the         * | ||||||
|  |  *   GNU General Public License for more details.                          * | ||||||
|  |  *                                                                         * | ||||||
|  |  *   You should have received a copy of the GNU General Public License     * | ||||||
|  |  *   along with awesome-widgets. If not, see http://www.gnu.org/licenses/  * | ||||||
|  |  ***************************************************************************/ | ||||||
|  |  | ||||||
|  |  | ||||||
|  | #ifndef AWTELEMETRYHANDLER_H | ||||||
|  | #define AWTELEMETRYHANDLER_H | ||||||
|  |  | ||||||
|  | #include <QObject> | ||||||
|  |  | ||||||
|  |  | ||||||
|  | class QNetworkReply; | ||||||
|  |  | ||||||
|  | class AWTelemetryHandler : public QObject | ||||||
|  | { | ||||||
|  |     Q_OBJECT | ||||||
|  |  | ||||||
|  | public: | ||||||
|  |     const char *REMOTE_TELEMETRY_URL = "https://arcanis.me/telemetry"; | ||||||
|  |  | ||||||
|  |     explicit AWTelemetryHandler(QObject *_parent = nullptr, const QString &_clientId = ""); | ||||||
|  |     ~AWTelemetryHandler() override; | ||||||
|  |     Q_INVOKABLE [[nodiscard]] QStringList get(const QString &_group) const; | ||||||
|  |     Q_INVOKABLE [[nodiscard]] QString getLast(const QString &_group) const; | ||||||
|  |     Q_INVOKABLE void init(int _count, bool _enableRemote, const QString &_clientId); | ||||||
|  |     Q_INVOKABLE [[nodiscard]] bool put(const QString &_group, const QString &_value) const; | ||||||
|  |     Q_INVOKABLE void uploadTelemetry(const QString &_group, const QString &_value); | ||||||
|  |  | ||||||
|  | signals: | ||||||
|  |     void replyReceived(const QString &_message); | ||||||
|  |  | ||||||
|  | private slots: | ||||||
|  |     void telemetryReplyRecieved(QNetworkReply *_reply); | ||||||
|  |  | ||||||
|  | private: | ||||||
|  |     static QString getKey(int _count); | ||||||
|  |     QString m_clientId; | ||||||
|  |     QString m_localFile; | ||||||
|  |     int m_storeCount = 0; | ||||||
|  |     bool m_uploadEnabled = false; | ||||||
|  | }; | ||||||
|  |  | ||||||
|  |  | ||||||
|  | #endif /* AWTELEMETRYHANDLER_H */ | ||||||
							
								
								
									
										174
									
								
								sources/awesome-widget/plugin/awupdatehelper.cpp
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										174
									
								
								sources/awesome-widget/plugin/awupdatehelper.cpp
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,174 @@ | |||||||
|  | /*************************************************************************** | ||||||
|  |  *   This file is part of awesome-widgets                                  * | ||||||
|  |  *                                                                         * | ||||||
|  |  *   awesome-widgets is free software: you can redistribute it and/or      * | ||||||
|  |  *   modify it under the terms of the GNU General Public License as        * | ||||||
|  |  *   published by the Free Software Foundation, either version 3 of the    * | ||||||
|  |  *   License, or (at your option) any later version.                       * | ||||||
|  |  *                                                                         * | ||||||
|  |  *   awesome-widgets is distributed in the hope that it will be useful,    * | ||||||
|  |  *   but WITHOUT ANY WARRANTY; without even the implied warranty of        * | ||||||
|  |  *   MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the         * | ||||||
|  |  *   GNU General Public License for more details.                          * | ||||||
|  |  *                                                                         * | ||||||
|  |  *   You should have received a copy of the GNU General Public License     * | ||||||
|  |  *   along with awesome-widgets. If not, see http://www.gnu.org/licenses/  * | ||||||
|  |  ***************************************************************************/ | ||||||
|  |  | ||||||
|  | #include "awupdatehelper.h" | ||||||
|  |  | ||||||
|  | #include <KI18n/KLocalizedString> | ||||||
|  |  | ||||||
|  | #include <QDesktopServices> | ||||||
|  | #include <QJsonDocument> | ||||||
|  | #include <QNetworkAccessManager> | ||||||
|  | #include <QNetworkReply> | ||||||
|  | #include <QSettings> | ||||||
|  |  | ||||||
|  | #include "awdebug.h" | ||||||
|  |  | ||||||
|  |  | ||||||
|  | AWUpdateHelper::AWUpdateHelper(QObject *_parent) | ||||||
|  |     : QObject(_parent) | ||||||
|  | { | ||||||
|  |     qCDebug(LOG_AW) << __PRETTY_FUNCTION__; | ||||||
|  |  | ||||||
|  |     m_foundVersion = QVersionNumber::fromString(VERSION); | ||||||
|  |     m_genericConfig = QString("%1/awesomewidgets/general.ini") | ||||||
|  |                           .arg(QStandardPaths::writableLocation(QStandardPaths::GenericDataLocation)); | ||||||
|  | } | ||||||
|  |  | ||||||
|  |  | ||||||
|  | AWUpdateHelper::~AWUpdateHelper() | ||||||
|  | { | ||||||
|  |     qCDebug(LOG_AW) << __PRETTY_FUNCTION__; | ||||||
|  | } | ||||||
|  |  | ||||||
|  |  | ||||||
|  | void AWUpdateHelper::checkUpdates(const bool _showAnyway) | ||||||
|  | { | ||||||
|  |     qCDebug(LOG_AW) << "Show anyway" << _showAnyway; | ||||||
|  |  | ||||||
|  |     // showAnyway options requires to show message if no updates found on direct | ||||||
|  |     // request. In case of automatic check no message will be shown | ||||||
|  |     auto *manager = new QNetworkAccessManager(nullptr); | ||||||
|  |     connect(manager, &QNetworkAccessManager::finished, | ||||||
|  |             [_showAnyway, this](QNetworkReply *reply) { return versionReplyRecieved(reply, _showAnyway); }); | ||||||
|  |  | ||||||
|  |     manager->get(QNetworkRequest(QUrl(VERSION_API))); | ||||||
|  | } | ||||||
|  |  | ||||||
|  |  | ||||||
|  | bool AWUpdateHelper::checkVersion() | ||||||
|  | { | ||||||
|  |     QSettings settings(m_genericConfig, QSettings::IniFormat); | ||||||
|  |     QVersionNumber version = QVersionNumber::fromString(settings.value("Version", QString(VERSION)).toString()); | ||||||
|  |     // update version | ||||||
|  |     settings.setValue("Version", QString(VERSION)); | ||||||
|  |     settings.sync(); | ||||||
|  |     qCInfo(LOG_AW) << "Found version" << version << "actual one is" << m_foundVersion; | ||||||
|  |  | ||||||
|  |     if ((version != m_foundVersion) && (!QString(CHANGELOG).isEmpty())) { | ||||||
|  |         genMessageBox(i18nc("Changelog of %1", VERSION), QString(CHANGELOG).replace('@', '\n'), QMessageBox::Ok) | ||||||
|  |             ->open(); | ||||||
|  |         return true; | ||||||
|  |     } else if (version != m_foundVersion) { | ||||||
|  |         qCWarning(LOG_AW) << "No changelog information provided"; | ||||||
|  |         return true; | ||||||
|  |     } else { | ||||||
|  |         qCInfo(LOG_AW) << "No need to update version"; | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     return false; | ||||||
|  | } | ||||||
|  |  | ||||||
|  |  | ||||||
|  | void AWUpdateHelper::showInfo(const QVersionNumber &_version) | ||||||
|  | { | ||||||
|  |     qCDebug(LOG_AW) << "Version" << _version; | ||||||
|  |  | ||||||
|  |     QString text = i18n("You are using the actual version %1", _version.toString()); | ||||||
|  |     if (!QString(COMMIT_SHA).isEmpty()) | ||||||
|  |         text += QString(" (%1)").arg(QString(COMMIT_SHA)); | ||||||
|  |     return genMessageBox(i18n("No new version found"), text, QMessageBox::Ok)->open(); | ||||||
|  | } | ||||||
|  |  | ||||||
|  |  | ||||||
|  | void AWUpdateHelper::showUpdates(const QVersionNumber &_version) | ||||||
|  | { | ||||||
|  |     qCDebug(LOG_AW) << "Version" << _version; | ||||||
|  |  | ||||||
|  |     QString text; | ||||||
|  |     text += i18nc("Current version : %1", VERSION); | ||||||
|  |     text += QString(COMMIT_SHA).isEmpty() ? "\n" : QString(" (%1)\n").arg(QString(COMMIT_SHA)); | ||||||
|  |     text += i18n("New version : %1", _version.toString()) + "\n\n"; | ||||||
|  |     text += i18n("Click \"Ok\" to download"); | ||||||
|  |  | ||||||
|  |     genMessageBox(i18n("There are updates"), text, QMessageBox::Ok | QMessageBox::Cancel) | ||||||
|  |         ->open(this, SLOT(userReplyOnUpdates(QAbstractButton *))); | ||||||
|  | } | ||||||
|  |  | ||||||
|  |  | ||||||
|  | void AWUpdateHelper::userReplyOnUpdates(QAbstractButton *_button) | ||||||
|  | { | ||||||
|  |     QMessageBox::ButtonRole ret = dynamic_cast<QMessageBox *>(sender())->buttonRole(_button); | ||||||
|  |     qCInfo(LOG_AW) << "User select" << ret; | ||||||
|  |  | ||||||
|  |     switch (ret) { | ||||||
|  |     case QMessageBox::AcceptRole: | ||||||
|  |         QDesktopServices::openUrl(QString(RELEASES) + m_foundVersion.toString()); | ||||||
|  |         break; | ||||||
|  |     case QMessageBox::RejectRole: | ||||||
|  |     default: | ||||||
|  |         break; | ||||||
|  |     } | ||||||
|  | } | ||||||
|  |  | ||||||
|  |  | ||||||
|  | void AWUpdateHelper::versionReplyRecieved(QNetworkReply *_reply, const bool _showAnyway) | ||||||
|  | { | ||||||
|  |     qCDebug(LOG_AW) << "Show message anyway" << _showAnyway; | ||||||
|  |     if (_reply->error() != QNetworkReply::NoError) { | ||||||
|  |         qCWarning(LOG_AW) << "An error occurs" << _reply->error() << "with message" << _reply->errorString(); | ||||||
|  |         return; | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     QJsonParseError error = QJsonParseError(); | ||||||
|  |     QJsonDocument jsonDoc = QJsonDocument::fromJson(_reply->readAll(), &error); | ||||||
|  |     if (error.error != QJsonParseError::NoError) { | ||||||
|  |         qCWarning(LOG_AW) << "Parse error" << error.errorString(); | ||||||
|  |         return; | ||||||
|  |     } | ||||||
|  |     _reply->deleteLater(); | ||||||
|  |  | ||||||
|  |     // convert to map | ||||||
|  |     QVariantMap firstRelease = jsonDoc.toVariant().toList().first().toMap(); | ||||||
|  |     QString version = firstRelease["tag_name"].toString(); | ||||||
|  |     version.remove("V."); | ||||||
|  |     m_foundVersion = QVersionNumber::fromString(version); | ||||||
|  |     qCInfo(LOG_AW) << "Update found version to" << m_foundVersion; | ||||||
|  |  | ||||||
|  |     QVersionNumber oldVersion = QVersionNumber::fromString(VERSION); | ||||||
|  |     if (oldVersion < m_foundVersion) | ||||||
|  |         return showUpdates(m_foundVersion); | ||||||
|  |     else if (_showAnyway) | ||||||
|  |         return showInfo(m_foundVersion); | ||||||
|  | } | ||||||
|  |  | ||||||
|  |  | ||||||
|  | // additional method which is used to show message box which does not block UI | ||||||
|  | QMessageBox *AWUpdateHelper::genMessageBox(const QString &_title, const QString &_body, | ||||||
|  |                                            const QMessageBox::StandardButtons _buttons) | ||||||
|  | { | ||||||
|  |     qCDebug(LOG_AW) << "Construct message box with title" << _title << "and body" << _body; | ||||||
|  |  | ||||||
|  |     auto *msgBox = new QMessageBox(nullptr); | ||||||
|  |     msgBox->setAttribute(Qt::WA_DeleteOnClose); | ||||||
|  |     msgBox->setModal(false); | ||||||
|  |     msgBox->setWindowTitle(_title); | ||||||
|  |     msgBox->setText(_body); | ||||||
|  |     msgBox->setStandardButtons(_buttons); | ||||||
|  |     msgBox->setIcon(QMessageBox::Information); | ||||||
|  |  | ||||||
|  |     return msgBox; | ||||||
|  | } | ||||||
							
								
								
									
										53
									
								
								sources/awesome-widget/plugin/awupdatehelper.h
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										53
									
								
								sources/awesome-widget/plugin/awupdatehelper.h
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,53 @@ | |||||||
|  | /*************************************************************************** | ||||||
|  |  *   This file is part of awesome-widgets                                  * | ||||||
|  |  *                                                                         * | ||||||
|  |  *   awesome-widgets is free software: you can redistribute it and/or      * | ||||||
|  |  *   modify it under the terms of the GNU General Public License as        * | ||||||
|  |  *   published by the Free Software Foundation, either version 3 of the    * | ||||||
|  |  *   License, or (at your option) any later version.                       * | ||||||
|  |  *                                                                         * | ||||||
|  |  *   awesome-widgets is distributed in the hope that it will be useful,    * | ||||||
|  |  *   but WITHOUT ANY WARRANTY; without even the implied warranty of        * | ||||||
|  |  *   MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the         * | ||||||
|  |  *   GNU General Public License for more details.                          * | ||||||
|  |  *                                                                         * | ||||||
|  |  *   You should have received a copy of the GNU General Public License     * | ||||||
|  |  *   along with awesome-widgets. If not, see http://www.gnu.org/licenses/  * | ||||||
|  |  ***************************************************************************/ | ||||||
|  |  | ||||||
|  |  | ||||||
|  | #ifndef AWUPDATEHELPER_H | ||||||
|  | #define AWUPDATEHELPER_H | ||||||
|  |  | ||||||
|  | #include <QMessageBox> | ||||||
|  | #include <QObject> | ||||||
|  | #include <QVersionNumber> | ||||||
|  |  | ||||||
|  |  | ||||||
|  | class QNetworkReply; | ||||||
|  |  | ||||||
|  | class AWUpdateHelper : public QObject | ||||||
|  | { | ||||||
|  |     Q_OBJECT | ||||||
|  |  | ||||||
|  | public: | ||||||
|  |     explicit AWUpdateHelper(QObject *_parent = nullptr); | ||||||
|  |     ~AWUpdateHelper() override; | ||||||
|  |     void checkUpdates(bool _showAnyway = false); | ||||||
|  |     bool checkVersion(); | ||||||
|  |  | ||||||
|  | private slots: | ||||||
|  |     static void showInfo(const QVersionNumber &_version); | ||||||
|  |     void showUpdates(const QVersionNumber &_version); | ||||||
|  |     void userReplyOnUpdates(QAbstractButton *_button); | ||||||
|  |     void versionReplyRecieved(QNetworkReply *_reply, bool _showAnyway); | ||||||
|  |  | ||||||
|  | private: | ||||||
|  |     static QMessageBox *genMessageBox(const QString &_title, const QString &_body, | ||||||
|  |                                       QMessageBox::StandardButtons _buttons); | ||||||
|  |     QVersionNumber m_foundVersion; | ||||||
|  |     QString m_genericConfig; | ||||||
|  | }; | ||||||
|  |  | ||||||
|  |  | ||||||
|  | #endif /* AWUPDATEHELPER_H */ | ||||||
| @ -5,8 +5,6 @@ add_definitions(-DTRANSLATION_DOMAIN=\"plasma_applet_org.kde.plasma.awesomewidge | |||||||
| include_directories( | include_directories( | ||||||
|         ${CMAKE_SOURCE_DIR} |         ${CMAKE_SOURCE_DIR} | ||||||
|         ${CMAKE_BINARY_DIR} |         ${CMAKE_BINARY_DIR} | ||||||
|         ${CMAKE_CURRENT_BINARY_DIR} |  | ||||||
|         ${CMAKE_CURRENT_BINARY_DIR}/../ |  | ||||||
|         ${PROJECT_TRDPARTY_DIR} |         ${PROJECT_TRDPARTY_DIR} | ||||||
|         ${Qt_INCLUDE} |         ${Qt_INCLUDE} | ||||||
|         ${Kf5_INCLUDE} |         ${Kf5_INCLUDE} | ||||||
| @ -15,9 +13,13 @@ include_directories( | |||||||
| file(GLOB SUBPROJECT_SOURCE *.cpp ${PROJECT_TRDPARTY_DIR}/qreplytimeout/*.cpp ${CMAKE_SOURCE_DIR}/*.cpp) | file(GLOB SUBPROJECT_SOURCE *.cpp ${PROJECT_TRDPARTY_DIR}/qreplytimeout/*.cpp ${CMAKE_SOURCE_DIR}/*.cpp) | ||||||
| file(GLOB SUBPROJECT_HEADER *.h ${PROJECT_TRDPARTY_DIR}/qreplytimeout/*.h) | file(GLOB SUBPROJECT_HEADER *.h ${PROJECT_TRDPARTY_DIR}/qreplytimeout/*.h) | ||||||
| file(GLOB SUBPROJECT_UI *.ui) | file(GLOB SUBPROJECT_UI *.ui) | ||||||
|  | file(GLOB SUBPROJECT_INI *.ini) | ||||||
|  | set(SUBPROJECT_CONFIGS ${CMAKE_CURRENT_SOURCE_DIR}/configs) | ||||||
|  | set(SUBPROJECT_FORMATTERS ${CMAKE_CURRENT_SOURCE_DIR}/formatters) | ||||||
| set(SUBPROJECT_GRAPHITEMS ${CMAKE_CURRENT_SOURCE_DIR}/desktops) | set(SUBPROJECT_GRAPHITEMS ${CMAKE_CURRENT_SOURCE_DIR}/desktops) | ||||||
| set(SUBPROJECT_QUOTES ${CMAKE_CURRENT_SOURCE_DIR}/quotes) | set(SUBPROJECT_QUOTES ${CMAKE_CURRENT_SOURCE_DIR}/quotes) | ||||||
| set(SUBPROJECT_SCRIPTS ${CMAKE_CURRENT_SOURCE_DIR}/scripts) | set(SUBPROJECT_SCRIPTS ${CMAKE_CURRENT_SOURCE_DIR}/scripts) | ||||||
|  | set(SUBPROJECT_REQUESTS ${CMAKE_CURRENT_SOURCE_DIR}/requests) | ||||||
| set(SUBPROJECT_UPGRADE ${CMAKE_CURRENT_SOURCE_DIR}/upgrade) | set(SUBPROJECT_UPGRADE ${CMAKE_CURRENT_SOURCE_DIR}/upgrade) | ||||||
| set(SUBPROJECT_WEATHER ${CMAKE_CURRENT_SOURCE_DIR}/weather) | set(SUBPROJECT_WEATHER ${CMAKE_CURRENT_SOURCE_DIR}/weather) | ||||||
| file(GLOB SUBPROJECT_WEATHER_JSON_IN *.json) | file(GLOB SUBPROJECT_WEATHER_JSON_IN *.json) | ||||||
| @ -31,9 +33,13 @@ add_library(${SUBPROJECT} STATIC ${SUBPROJECT_SOURCE} ${SUBPROJECT_HEADER} ${SUB | |||||||
| target_link_libraries(${SUBPROJECT} ${Qt_LIBRARIES} ${Kf5_LIBRARIES}) | target_link_libraries(${SUBPROJECT} ${Qt_LIBRARIES} ${Kf5_LIBRARIES}) | ||||||
|  |  | ||||||
| # install | # install | ||||||
|  | install(DIRECTORY ${SUBPROJECT_CONFIGS} DESTINATION ${DATA_INSTALL_DIR}/${PROJECT_NAME}) | ||||||
|  | install(DIRECTORY ${SUBPROJECT_FORMATTERS} DESTINATION ${DATA_INSTALL_DIR}/${PROJECT_NAME}) | ||||||
| install(DIRECTORY ${SUBPROJECT_GRAPHITEMS} DESTINATION ${DATA_INSTALL_DIR}/${PROJECT_NAME}) | install(DIRECTORY ${SUBPROJECT_GRAPHITEMS} DESTINATION ${DATA_INSTALL_DIR}/${PROJECT_NAME}) | ||||||
| install(DIRECTORY ${SUBPROJECT_QUOTES} DESTINATION ${DATA_INSTALL_DIR}/${PROJECT_NAME}) | install(DIRECTORY ${SUBPROJECT_QUOTES} DESTINATION ${DATA_INSTALL_DIR}/${PROJECT_NAME}) | ||||||
| install(DIRECTORY ${SUBPROJECT_SCRIPTS} DESTINATION ${DATA_INSTALL_DIR}/${PROJECT_NAME}) | install(DIRECTORY ${SUBPROJECT_SCRIPTS} DESTINATION ${DATA_INSTALL_DIR}/${PROJECT_NAME}) | ||||||
|  | install(DIRECTORY ${SUBPROJECT_REQUESTS} DESTINATION ${DATA_INSTALL_DIR}/${PROJECT_NAME}) | ||||||
| install(DIRECTORY ${SUBPROJECT_UPGRADE} DESTINATION ${DATA_INSTALL_DIR}/${PROJECT_NAME}) | install(DIRECTORY ${SUBPROJECT_UPGRADE} DESTINATION ${DATA_INSTALL_DIR}/${PROJECT_NAME}) | ||||||
| install(DIRECTORY ${SUBPROJECT_WEATHER} DESTINATION ${DATA_INSTALL_DIR}/${PROJECT_NAME}) | install(DIRECTORY ${SUBPROJECT_WEATHER} DESTINATION ${DATA_INSTALL_DIR}/${PROJECT_NAME}) | ||||||
|  | install(FILES ${SUBPROJECT_INI} DESTINATION ${CONFIG_INSTALL_DIR}) | ||||||
| install(FILES ${CMAKE_CURRENT_BINARY_DIR}/${SUBPROJECT_WEATHER_JSON} DESTINATION ${DATA_INSTALL_DIR}/${PROJECT_NAME}/weather) | install(FILES ${CMAKE_CURRENT_BINARY_DIR}/${SUBPROJECT_WEATHER_JSON} DESTINATION ${DATA_INSTALL_DIR}/${PROJECT_NAME}/weather) | ||||||
|  | |||||||
| @ -18,23 +18,22 @@ | |||||||
| #include "abstractextitem.h" | #include "abstractextitem.h" | ||||||
|  |  | ||||||
| #include <QDir> | #include <QDir> | ||||||
|  | #include <QLocalServer> | ||||||
|  | #include <QRandomGenerator> | ||||||
| #include <QSettings> | #include <QSettings> | ||||||
| #include <QStandardPaths> | #include <QStandardPaths> | ||||||
| #include <QTime> |  | ||||||
|  |  | ||||||
| #include "awdebug.h" | #include "abstractextitemaggregator.h" | ||||||
| #include "version.h" | #include "qcronscheduler.h" | ||||||
|  |  | ||||||
|  |  | ||||||
| AbstractExtItem::AbstractExtItem(QWidget *parent, const QString desktopName, | AbstractExtItem::AbstractExtItem(QWidget *_parent, const QString &_filePath) | ||||||
|                                  const QStringList directories) |     : QDialog(_parent) | ||||||
|     : QDialog(parent) |     , m_fileName(_filePath) | ||||||
|     , m_fileName(desktopName) |  | ||||||
|     , m_dirs(directories) |  | ||||||
| { | { | ||||||
|     qCDebug(LOG_LIB) << __PRETTY_FUNCTION__; |     qCDebug(LOG_LIB) << __PRETTY_FUNCTION__; | ||||||
|     qCDebug(LOG_LIB) << "Desktop name" << desktopName; |  | ||||||
|     qCDebug(LOG_LIB) << "Directories" << directories; |     qCDebug(LOG_LIB) << "Desktop name" << _filePath; | ||||||
|  |  | ||||||
|     m_name = m_fileName; |     m_name = m_fileName; | ||||||
| } | } | ||||||
| @ -43,13 +42,70 @@ AbstractExtItem::AbstractExtItem(QWidget *parent, const QString desktopName, | |||||||
| AbstractExtItem::~AbstractExtItem() | AbstractExtItem::~AbstractExtItem() | ||||||
| { | { | ||||||
|     qCDebug(LOG_LIB) << __PRETTY_FUNCTION__; |     qCDebug(LOG_LIB) << __PRETTY_FUNCTION__; | ||||||
|  |  | ||||||
|  |     if (m_socket) { | ||||||
|  |         m_socket->close(); | ||||||
|  |         QLocalServer::removeServer(socket()); | ||||||
|  |         m_socket->deleteLater(); | ||||||
|  |     } | ||||||
| } | } | ||||||
|  |  | ||||||
|  |  | ||||||
| template <class T> T *AbstractExtItem::copy(const QString, const int) | void AbstractExtItem::bumpApi(const int _newVer) | ||||||
| { | { | ||||||
|     // an analog of pure virtual method |     qCDebug(LOG_LIB) << "Bump API using new version" << _newVer; | ||||||
|     return new T(); |  | ||||||
|  |     // update for current API | ||||||
|  |     if ((apiVersion() > 0) && (apiVersion() < _newVer)) { | ||||||
|  |         qCWarning(LOG_LIB) << "Bump API version from" << apiVersion() << "to" << _newVer; | ||||||
|  |         setApiVersion(_newVer); | ||||||
|  |         writeConfiguration(); | ||||||
|  |     } | ||||||
|  | } | ||||||
|  |  | ||||||
|  |  | ||||||
|  | void AbstractExtItem::copyDefaults(AbstractExtItem *_other) const | ||||||
|  | { | ||||||
|  |     _other->setActive(isActive()); | ||||||
|  |     _other->setApiVersion(apiVersion()); | ||||||
|  |     _other->setComment(comment()); | ||||||
|  |     _other->setCron(cron()); | ||||||
|  |     _other->setInterval(interval()); | ||||||
|  |     _other->setName(name()); | ||||||
|  |     _other->setSocket(socket()); | ||||||
|  | } | ||||||
|  |  | ||||||
|  |  | ||||||
|  | void AbstractExtItem::startTimer() | ||||||
|  | { | ||||||
|  |     if (!socket().isEmpty()) | ||||||
|  |         // check if there is active socket setup | ||||||
|  |         return; | ||||||
|  |     else if (!cron().isEmpty()) | ||||||
|  |         // check if there is active scheduler | ||||||
|  |         return; | ||||||
|  |     else if (m_times == 1) | ||||||
|  |         // check if it is time to update | ||||||
|  |         emit(requestDataUpdate()); | ||||||
|  |  | ||||||
|  |     // update counter value | ||||||
|  |     if (m_times >= interval()) | ||||||
|  |         m_times = 0; | ||||||
|  |     m_times++; | ||||||
|  | } | ||||||
|  |  | ||||||
|  |  | ||||||
|  | QString AbstractExtItem::writtableConfig() const | ||||||
|  | { | ||||||
|  |     QString path = m_fileName; | ||||||
|  |     QString name = QFileInfo(path).fileName(); | ||||||
|  |     path.remove(path.count() - name.count() - 1, name.count() + 1); | ||||||
|  |     QString dir = QFileInfo(path).fileName(); | ||||||
|  |  | ||||||
|  |     return QString("%1/awesomewidgets/%2/%3") | ||||||
|  |         .arg(QStandardPaths::writableLocation(QStandardPaths::GenericDataLocation)) | ||||||
|  |         .arg(dir) | ||||||
|  |         .arg(name); | ||||||
| } | } | ||||||
|  |  | ||||||
|  |  | ||||||
| @ -65,9 +121,9 @@ QString AbstractExtItem::comment() const | |||||||
| } | } | ||||||
|  |  | ||||||
|  |  | ||||||
| QStringList AbstractExtItem::directories() const | QString AbstractExtItem::cron() const | ||||||
| { | { | ||||||
|     return m_dirs; |     return m_cron; | ||||||
| } | } | ||||||
|  |  | ||||||
|  |  | ||||||
| @ -101,11 +157,17 @@ int AbstractExtItem::number() const | |||||||
| } | } | ||||||
|  |  | ||||||
|  |  | ||||||
| QString AbstractExtItem::tag(const QString _type) const | QString AbstractExtItem::socket() const | ||||||
|  | { | ||||||
|  |     return m_socketFile; | ||||||
|  | } | ||||||
|  |  | ||||||
|  |  | ||||||
|  | QString AbstractExtItem::tag(const QString &_type) const | ||||||
| { | { | ||||||
|     qCDebug(LOG_LIB) << "Tag type" << _type; |     qCDebug(LOG_LIB) << "Tag type" << _type; | ||||||
|  |  | ||||||
|     return QString("%1%2").arg(_type).arg(m_number); |     return QString("%1%2").arg(_type).arg(number()); | ||||||
| } | } | ||||||
|  |  | ||||||
|  |  | ||||||
| @ -125,7 +187,7 @@ void AbstractExtItem::setActive(const bool _state) | |||||||
| } | } | ||||||
|  |  | ||||||
|  |  | ||||||
| void AbstractExtItem::setComment(const QString _comment) | void AbstractExtItem::setComment(const QString &_comment) | ||||||
| { | { | ||||||
|     qCDebug(LOG_LIB) << "Comment" << _comment; |     qCDebug(LOG_LIB) << "Comment" << _comment; | ||||||
|  |  | ||||||
| @ -133,6 +195,26 @@ void AbstractExtItem::setComment(const QString _comment) | |||||||
| } | } | ||||||
|  |  | ||||||
|  |  | ||||||
|  | void AbstractExtItem::setCron(const QString &_cron) | ||||||
|  | { | ||||||
|  |     qCDebug(LOG_LIB) << "Cron string" << _cron; | ||||||
|  |     // deinit module first | ||||||
|  |     if (m_scheduler) { | ||||||
|  |         disconnect(m_scheduler, SIGNAL(activated()), this, SIGNAL(requestDataUpdate())); | ||||||
|  |         delete m_scheduler; | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     m_cron = _cron; | ||||||
|  |     if (cron().isEmpty()) | ||||||
|  |         return; | ||||||
|  |  | ||||||
|  |     // init scheduler | ||||||
|  |     m_scheduler = new QCronScheduler(this); | ||||||
|  |     m_scheduler->parse(cron()); | ||||||
|  |     connect(m_scheduler, SIGNAL(activated()), this, SIGNAL(requestDataUpdate())); | ||||||
|  | } | ||||||
|  |  | ||||||
|  |  | ||||||
| void AbstractExtItem::setInterval(const int _interval) | void AbstractExtItem::setInterval(const int _interval) | ||||||
| { | { | ||||||
|     qCDebug(LOG_LIB) << "Interval" << _interval; |     qCDebug(LOG_LIB) << "Interval" << _interval; | ||||||
| @ -143,7 +225,7 @@ void AbstractExtItem::setInterval(const int _interval) | |||||||
| } | } | ||||||
|  |  | ||||||
|  |  | ||||||
| void AbstractExtItem::setName(const QString _name) | void AbstractExtItem::setName(const QString &_name) | ||||||
| { | { | ||||||
|     qCDebug(LOG_LIB) << "Name" << _name; |     qCDebug(LOG_LIB) << "Name" << _name; | ||||||
|  |  | ||||||
| @ -154,74 +236,104 @@ void AbstractExtItem::setName(const QString _name) | |||||||
| void AbstractExtItem::setNumber(int _number) | void AbstractExtItem::setNumber(int _number) | ||||||
| { | { | ||||||
|     qCDebug(LOG_LIB) << "Number" << _number; |     qCDebug(LOG_LIB) << "Number" << _number; | ||||||
|     if (_number == -1) |     bool generateNumber = (_number == -1); | ||||||
|  |     if (generateNumber) { | ||||||
|         _number = []() { |         _number = []() { | ||||||
|             qCWarning(LOG_LIB) << "Number is empty, generate new one"; |             qCWarning(LOG_LIB) << "Number is empty, generate new one"; | ||||||
|             qsrand(QTime::currentTime().msec()); |             auto n = QRandomGenerator::global()->generate() % 1000; | ||||||
|             int n = qrand() % 1000; |  | ||||||
|             qCInfo(LOG_LIB) << "Generated number is" << n; |             qCInfo(LOG_LIB) << "Generated number is" << n; | ||||||
|             return n; |             return n; | ||||||
|         }(); |         }(); | ||||||
|  |     } | ||||||
|  |  | ||||||
|     m_number = _number; |     m_number = _number; | ||||||
|  |     if (generateNumber) | ||||||
|  |         writeConfiguration(); | ||||||
|  | } | ||||||
|  |  | ||||||
|  |  | ||||||
|  | void AbstractExtItem::setSocket(const QString &_socket) | ||||||
|  | { | ||||||
|  |     qCDebug(LOG_LIB) << "Socket" << _socket; | ||||||
|  |     // remove old socket first | ||||||
|  |     deinitSocket(); | ||||||
|  |  | ||||||
|  |     m_socketFile = _socket; | ||||||
|  | } | ||||||
|  |  | ||||||
|  |  | ||||||
|  | void AbstractExtItem::deinitSocket() | ||||||
|  | { | ||||||
|  |     if (!m_socket) | ||||||
|  |         return; | ||||||
|  |  | ||||||
|  |     m_socket->close(); | ||||||
|  |     QLocalServer::removeServer(socket()); | ||||||
|  |     delete m_socket; | ||||||
|  |     disconnect(m_socket, SIGNAL(newConnection()), this, SLOT(newConnectionReceived())); | ||||||
|  | } | ||||||
|  |  | ||||||
|  |  | ||||||
|  | void AbstractExtItem::initSocket() | ||||||
|  | { | ||||||
|  |     // remove old socket first | ||||||
|  |     deinitSocket(); | ||||||
|  |  | ||||||
|  |     m_socket = new QLocalServer(this); | ||||||
|  |     bool listening = m_socket->listen(socket()); | ||||||
|  |     qCInfo(LOG_LIB) << "Server listening on" << socket() << listening; | ||||||
|  |     connect(m_socket, SIGNAL(newConnection()), this, SLOT(newConnectionReceived())); | ||||||
| } | } | ||||||
|  |  | ||||||
|  |  | ||||||
| void AbstractExtItem::readConfiguration() | void AbstractExtItem::readConfiguration() | ||||||
| { | { | ||||||
|     for (int i = m_dirs.count() - 1; i >= 0; i--) { |     QSettings settings(m_fileName, QSettings::IniFormat); | ||||||
|         if (!QDir(m_dirs.at(i)).entryList(QDir::Files).contains(m_fileName)) |  | ||||||
|             continue; |  | ||||||
|         QSettings settings(QString("%1/%2").arg(m_dirs.at(i)).arg(m_fileName), |  | ||||||
|                            QSettings::IniFormat); |  | ||||||
|  |  | ||||||
|         settings.beginGroup(QString("Desktop Entry")); |     settings.beginGroup("Desktop Entry"); | ||||||
|         setName(settings.value(QString("Name"), m_name).toString()); |     setName(settings.value("Name", name()).toString()); | ||||||
|         setComment(settings.value(QString("Comment"), m_comment).toString()); |     setComment(settings.value("Comment", comment()).toString()); | ||||||
|         setApiVersion( |     setApiVersion(settings.value("X-AW-ApiVersion", apiVersion()).toInt()); | ||||||
|             settings.value(QString("X-AW-ApiVersion"), m_apiVersion).toInt()); |     setActive(settings.value("X-AW-Active", isActive()).toBool()); | ||||||
|         setActive(settings.value(QString("X-AW-Active"), QVariant(m_active)) |     setInterval(settings.value("X-AW-Interval", interval()).toInt()); | ||||||
|                       .toString() |     setNumber(settings.value("X-AW-Number", number()).toInt()); | ||||||
|                   == QString("true")); |     setCron(settings.value("X-AW-Schedule", cron()).toString()); | ||||||
|         setInterval( |     setSocket(settings.value("X-AW-Socket", socket()).toString()); | ||||||
|             settings.value(QString("X-AW-Interval"), m_interval).toInt()); |  | ||||||
|         setNumber(settings.value(QString("X-AW-Number"), m_number).toInt()); |  | ||||||
|     settings.endGroup(); |     settings.endGroup(); | ||||||
|     } |  | ||||||
| } | } | ||||||
|  |  | ||||||
|  |  | ||||||
| bool AbstractExtItem::tryDelete() const | bool AbstractExtItem::tryDelete() const | ||||||
| { | { | ||||||
|     foreach (QString dir, m_dirs) { |     bool status = QFile::remove(m_fileName); | ||||||
|         bool status = QFile::remove(QString("%1/%2").arg(dir).arg(m_fileName)); |     qCInfo(LOG_LIB) << "Remove file" << m_fileName << status; | ||||||
|         qCInfo(LOG_LIB) << "Remove file" |  | ||||||
|                         << QString("%1/%2").arg(dir).arg(m_fileName) << status; |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     // check if exists |     return status; | ||||||
|     foreach (QString dir, m_dirs) |  | ||||||
|         if (QFile::exists(QString("%1/%2").arg(dir).arg(m_fileName))) |  | ||||||
|             return false; |  | ||||||
|     return true; |  | ||||||
| } | } | ||||||
|  |  | ||||||
|  |  | ||||||
| void AbstractExtItem::writeConfiguration() const | void AbstractExtItem::writeConfiguration() const | ||||||
| { | { | ||||||
|     QSettings settings(QString("%1/%2").arg(m_dirs.first()).arg(m_fileName), |     QSettings settings(writtableConfig(), QSettings::IniFormat); | ||||||
|                        QSettings::IniFormat); |  | ||||||
|     qCInfo(LOG_LIB) << "Configuration file" << settings.fileName(); |     qCInfo(LOG_LIB) << "Configuration file" << settings.fileName(); | ||||||
|  |  | ||||||
|     settings.beginGroup(QString("Desktop Entry")); |     settings.beginGroup("Desktop Entry"); | ||||||
|     settings.setValue(QString("Encoding"), QString("UTF-8")); |     settings.setValue("Encoding", "UTF-8"); | ||||||
|     settings.setValue(QString("Name"), m_name); |     settings.setValue("Name", name()); | ||||||
|     settings.setValue(QString("Comment"), m_comment); |     settings.setValue("Comment", comment()); | ||||||
|     settings.setValue(QString("X-AW-ApiVersion"), m_apiVersion); |     settings.setValue("X-AW-ApiVersion", apiVersion()); | ||||||
|     settings.setValue(QString("X-AW-Active"), QVariant(m_active).toString()); |     settings.setValue("X-AW-Active", isActive()); | ||||||
|     settings.setValue(QString("X-AW-Interval"), m_interval); |     settings.setValue("X-AW-Interval", interval()); | ||||||
|     settings.setValue(QString("X-AW-Number"), m_number); |     settings.setValue("X-AW-Number", number()); | ||||||
|  |     settings.setValue("X-AW-Schedule", cron()); | ||||||
|  |     settings.setValue("X-AW-Socket", socket()); | ||||||
|     settings.endGroup(); |     settings.endGroup(); | ||||||
|  |  | ||||||
|     settings.sync(); |     settings.sync(); | ||||||
| } | } | ||||||
|  |  | ||||||
|  |  | ||||||
|  | void AbstractExtItem::newConnectionReceived() | ||||||
|  | { | ||||||
|  |     emit(requestDataUpdate()); | ||||||
|  | } | ||||||
|  | |||||||
| @ -22,62 +22,84 @@ | |||||||
| #include <QVariant> | #include <QVariant> | ||||||
|  |  | ||||||
|  |  | ||||||
|  | class QCronScheduler; | ||||||
|  | class QLocalServer; | ||||||
|  |  | ||||||
| class AbstractExtItem : public QDialog | class AbstractExtItem : public QDialog | ||||||
| { | { | ||||||
|     Q_OBJECT |     Q_OBJECT | ||||||
|     Q_PROPERTY(bool active READ isActive WRITE setActive) |     Q_PROPERTY(bool active READ isActive WRITE setActive) | ||||||
|     Q_PROPERTY(int apiVersion READ apiVersion WRITE setApiVersion) |     Q_PROPERTY(int apiVersion READ apiVersion WRITE setApiVersion) | ||||||
|     Q_PROPERTY(QString comment READ comment WRITE setComment) |     Q_PROPERTY(QString comment READ comment WRITE setComment) | ||||||
|     Q_PROPERTY(QStringList directories READ directories) |     Q_PROPERTY(QString cron READ cron WRITE setCron) | ||||||
|     Q_PROPERTY(QString fileName READ fileName) |     Q_PROPERTY(QString fileName READ fileName) | ||||||
|     Q_PROPERTY(int interval READ interval WRITE setInterval) |     Q_PROPERTY(int interval READ interval WRITE setInterval) | ||||||
|     Q_PROPERTY(QString name READ name WRITE setName) |     Q_PROPERTY(QString name READ name WRITE setName) | ||||||
|     Q_PROPERTY(int number READ number WRITE setNumber) |     Q_PROPERTY(int number READ number WRITE setNumber) | ||||||
|  |     Q_PROPERTY(QString socket READ socket WRITE setSocket) | ||||||
|     Q_PROPERTY(QString uniq READ uniq) |     Q_PROPERTY(QString uniq READ uniq) | ||||||
|  |  | ||||||
| public: | public: | ||||||
|     explicit AbstractExtItem(QWidget *parent = nullptr, |     explicit AbstractExtItem(QWidget *_parent = nullptr, const QString &_filePath = ""); | ||||||
|                              const QString desktopName = QString(), |     ~AbstractExtItem() override; | ||||||
|                              const QStringList directories = QStringList()); |     virtual void bumpApi(int _newVer); | ||||||
|     virtual ~AbstractExtItem(); |     virtual AbstractExtItem *copy(const QString &_fileName, int _number) = 0; | ||||||
|     template <class T> T *copy(const QString, const int); |     virtual void copyDefaults(AbstractExtItem *_other) const; | ||||||
|  |     virtual void startTimer(); | ||||||
|  |     [[nodiscard]] QString writtableConfig() const; | ||||||
|     // get methods |     // get methods | ||||||
|     int apiVersion() const; |     [[nodiscard]] int apiVersion() const; | ||||||
|     QString comment() const; |     [[nodiscard]] QString comment() const; | ||||||
|     QStringList directories() const; |     [[nodiscard]] QString cron() const; | ||||||
|     QString fileName() const; |     [[nodiscard]] QString fileName() const; | ||||||
|     int interval() const; |     [[nodiscard]] int interval() const; | ||||||
|     bool isActive() const; |     [[nodiscard]] bool isActive() const; | ||||||
|     QString name() const; |     [[nodiscard]] QString name() const; | ||||||
|     int number() const; |     [[nodiscard]] int number() const; | ||||||
|     QString tag(const QString _type) const; |     [[nodiscard]] QString socket() const; | ||||||
|     virtual QString uniq() const = 0; |     [[nodiscard]] QString tag(const QString &_type) const; | ||||||
|  |     [[nodiscard]] virtual QString uniq() const = 0; | ||||||
|     // set methods |     // set methods | ||||||
|     void setApiVersion(const int _apiVersion = 0); |     void setApiVersion(int _apiVersion); | ||||||
|     void setActive(const bool _state = true); |     void setActive(bool _state); | ||||||
|     void setComment(const QString _comment = QString("empty")); |     void setComment(const QString &_comment); | ||||||
|     void setInterval(const int _interval = 1); |     void setCron(const QString &_cron); | ||||||
|     void setName(const QString _name = QString("none")); |     void setInterval(int _interval); | ||||||
|     void setNumber(int _number = -1); |     void setName(const QString &_name); | ||||||
|  |     void setNumber(int _number); | ||||||
|  |     void setSocket(const QString &_socket); | ||||||
|  |  | ||||||
|  | signals: | ||||||
|  |     void dataReceived(const QVariantHash &_data); | ||||||
|  |     void requestDataUpdate(); | ||||||
|  |  | ||||||
| public slots: | public slots: | ||||||
|  |     virtual void deinitSocket(); | ||||||
|  |     virtual void initSocket(); | ||||||
|     virtual void readConfiguration(); |     virtual void readConfiguration(); | ||||||
|     virtual QVariantHash run() = 0; |     virtual QVariantHash run() = 0; | ||||||
|     virtual int showConfiguration(const QVariant args = QVariant()) = 0; |     virtual int showConfiguration(const QVariant &_args) = 0; | ||||||
|     bool tryDelete() const; |     [[nodiscard]] virtual bool tryDelete() const; | ||||||
|     virtual void writeConfiguration() const; |     virtual void writeConfiguration() const; | ||||||
|  |  | ||||||
|  | private slots: | ||||||
|  |     void newConnectionReceived(); | ||||||
|  |  | ||||||
| private: | private: | ||||||
|     QString m_fileName; |     QCronScheduler *m_scheduler = nullptr; | ||||||
|     QStringList m_dirs; |     QString m_fileName = "/dev/null"; | ||||||
|  |     int m_times = 0; | ||||||
|     virtual void translate() = 0; |     virtual void translate() = 0; | ||||||
|     // properties |     // properties | ||||||
|     int m_apiVersion = 0; |     int m_apiVersion = 0; | ||||||
|     bool m_active = true; |     bool m_active = true; | ||||||
|     QString m_comment = QString("empty"); |     QString m_comment = "empty"; | ||||||
|  |     QString m_cron = ""; | ||||||
|     int m_interval = 1; |     int m_interval = 1; | ||||||
|     QString m_name = QString("none"); |     QString m_name = "none"; | ||||||
|     int m_number = -1; |     int m_number = -1; | ||||||
|  |     QLocalServer *m_socket = nullptr; | ||||||
|  |     QString m_socketFile = ""; | ||||||
| }; | }; | ||||||
|  |  | ||||||
|  |  | ||||||
|  | |||||||
| @ -16,41 +16,39 @@ | |||||||
|  ***************************************************************************/ |  ***************************************************************************/ | ||||||
|  |  | ||||||
| #include "abstractextitemaggregator.h" | #include "abstractextitemaggregator.h" | ||||||
|  | #include "ui_abstractextitemaggregator.h" | ||||||
|  |  | ||||||
| #include <KI18n/KLocalizedString> | #include <KI18n/KLocalizedString> | ||||||
|  |  | ||||||
| #include <QHBoxLayout> | #include <QDir> | ||||||
| #include <QInputDialog> | #include <QInputDialog> | ||||||
| #include <QLineEdit> | #include <QPushButton> | ||||||
|  | #include <utility> | ||||||
| #include "awdebug.h" |  | ||||||
|  |  | ||||||
|  |  | ||||||
| AbstractExtItemAggregator::AbstractExtItemAggregator(QWidget *parent) | AbstractExtItemAggregator::AbstractExtItemAggregator(QWidget *_parent, QString _type) | ||||||
|     : QWidget(parent) |     : QDialog(_parent) | ||||||
|  |     , ui(new Ui::AbstractExtItemAggregator) | ||||||
|  |     , m_type(std::move(_type)) | ||||||
| { | { | ||||||
|     qCDebug(LOG_LIB) << __PRETTY_FUNCTION__; |     qCDebug(LOG_LIB) << __PRETTY_FUNCTION__; | ||||||
|  |  | ||||||
|     dialog = new QDialog(this); |     // create directory at $HOME | ||||||
|     widgetDialog = new QListWidget(dialog); |     QString localDir = QString("%1/awesomewidgets/%2") | ||||||
|     dialogButtons = new QDialogButtonBox( |                            .arg(QStandardPaths::writableLocation(QStandardPaths::GenericDataLocation)) | ||||||
|         QDialogButtonBox::Open | QDialogButtonBox::Close, Qt::Vertical, dialog); |                            .arg(type()); | ||||||
|     copyButton |     QDir localDirectory; | ||||||
|         = dialogButtons->addButton(i18n("Copy"), QDialogButtonBox::ActionRole); |     if (localDirectory.mkpath(localDir)) | ||||||
|     createButton = dialogButtons->addButton(i18n("Create"), |         qCInfo(LOG_LIB) << "Created directory" << localDir; | ||||||
|                                             QDialogButtonBox::ActionRole); |  | ||||||
|     deleteButton = dialogButtons->addButton(i18n("Remove"), |  | ||||||
|                                             QDialogButtonBox::ActionRole); |  | ||||||
|     QHBoxLayout *layout = new QHBoxLayout(dialog); |  | ||||||
|     layout->addWidget(widgetDialog); |  | ||||||
|     layout->addWidget(dialogButtons); |  | ||||||
|     dialog->setLayout(layout); |  | ||||||
|  |  | ||||||
|     connect(dialogButtons, SIGNAL(clicked(QAbstractButton *)), this, |     ui->setupUi(this); | ||||||
|             SLOT(editItemButtonPressed(QAbstractButton *))); |     copyButton = ui->buttonBox->addButton(i18n("Copy"), QDialogButtonBox::ActionRole); | ||||||
|     connect(dialogButtons, SIGNAL(rejected()), dialog, SLOT(reject())); |     createButton = ui->buttonBox->addButton(i18n("Create"), QDialogButtonBox::ActionRole); | ||||||
|     connect(widgetDialog, SIGNAL(itemActivated(QListWidgetItem *)), this, |     deleteButton = ui->buttonBox->addButton(i18n("Remove"), QDialogButtonBox::ActionRole); | ||||||
|             SLOT(editItemActivated(QListWidgetItem *))); |  | ||||||
|  |     connect(ui->buttonBox, SIGNAL(clicked(QAbstractButton *)), this, SLOT(editItemButtonPressed(QAbstractButton *))); | ||||||
|  |     connect(ui->buttonBox, SIGNAL(rejected()), this, SLOT(reject())); | ||||||
|  |     connect(ui->listWidget, SIGNAL(itemActivated(QListWidgetItem *)), this, SLOT(editItemActivated(QListWidgetItem *))); | ||||||
| } | } | ||||||
|  |  | ||||||
|  |  | ||||||
| @ -58,32 +56,147 @@ AbstractExtItemAggregator::~AbstractExtItemAggregator() | |||||||
| { | { | ||||||
|     qCDebug(LOG_LIB) << __PRETTY_FUNCTION__; |     qCDebug(LOG_LIB) << __PRETTY_FUNCTION__; | ||||||
|  |  | ||||||
|     delete dialog; |     delete ui; | ||||||
|  | } | ||||||
|  |  | ||||||
|  |  | ||||||
|  | void AbstractExtItemAggregator::copyItem() | ||||||
|  | { | ||||||
|  |     AbstractExtItem *source = itemFromWidget(); | ||||||
|  |     QString fileName = getName(); | ||||||
|  |     int number = uniqNumber(); | ||||||
|  |     QString dir = QString("%1/awesomewidgets/%2") | ||||||
|  |                       .arg(QStandardPaths::writableLocation(QStandardPaths::GenericDataLocation)) | ||||||
|  |                       .arg(m_type); | ||||||
|  |     if ((!source) || (fileName.isEmpty())) { | ||||||
|  |         qCWarning(LOG_LIB) << "Nothing to copy"; | ||||||
|  |         return; | ||||||
|  |     } | ||||||
|  |     QString filePath = QString("%1/%2").arg(dir).arg(fileName); | ||||||
|  |  | ||||||
|  |     AbstractExtItem *newItem = source->copy(filePath, number); | ||||||
|  |     if (newItem->showConfiguration(configArgs()) == 1) { | ||||||
|  |         initItems(); | ||||||
|  |         repaintList(); | ||||||
|  |     } | ||||||
|  | } | ||||||
|  |  | ||||||
|  |  | ||||||
|  | void AbstractExtItemAggregator::deleteItem() | ||||||
|  | { | ||||||
|  |     AbstractExtItem *source = itemFromWidget(); | ||||||
|  |     if (!source) { | ||||||
|  |         qCWarning(LOG_LIB) << "Nothing to delete"; | ||||||
|  |         return; | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     if (source->tryDelete()) { | ||||||
|  |         initItems(); | ||||||
|  |         repaintList(); | ||||||
|  |     } | ||||||
|  | } | ||||||
|  |  | ||||||
|  |  | ||||||
|  | void AbstractExtItemAggregator::editItem() | ||||||
|  | { | ||||||
|  |     AbstractExtItem *source = itemFromWidget(); | ||||||
|  |     if (!source) { | ||||||
|  |         qCWarning(LOG_LIB) << "Nothing to edit"; | ||||||
|  |         return; | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     if (source->showConfiguration(configArgs()) == 1) { | ||||||
|  |         initItems(); | ||||||
|  |         repaintList(); | ||||||
|  |     } | ||||||
| } | } | ||||||
|  |  | ||||||
|  |  | ||||||
| QString AbstractExtItemAggregator::getName() | QString AbstractExtItemAggregator::getName() | ||||||
| { | { | ||||||
|     bool ok; |     bool ok; | ||||||
|     QString name = QInputDialog::getText(this, i18n("Enter file name"), |     QString name = QInputDialog::getText(this, i18n("Enter file name"), i18n("File name"), QLineEdit::Normal, "", &ok); | ||||||
|                                          i18n("File name"), QLineEdit::Normal, |  | ||||||
|                                          QString(""), &ok); |  | ||||||
|     if ((!ok) || (name.isEmpty())) |     if ((!ok) || (name.isEmpty())) | ||||||
|         return QString(""); |         return ""; | ||||||
|     if (!name.endsWith(QString(".desktop"))) |     if (!name.endsWith(".desktop")) | ||||||
|         name += QString(".desktop"); |         name += ".desktop"; | ||||||
|  |  | ||||||
|     return name; |     return name; | ||||||
| } | } | ||||||
|  |  | ||||||
|  |  | ||||||
|  | AbstractExtItem *AbstractExtItemAggregator::itemFromWidget() const | ||||||
|  | { | ||||||
|  |     QListWidgetItem *widgetItem = ui->listWidget->currentItem(); | ||||||
|  |     if (!widgetItem) | ||||||
|  |         return nullptr; | ||||||
|  |  | ||||||
|  |     AbstractExtItem *found = nullptr; | ||||||
|  |     for (auto &item : items()) { | ||||||
|  |         QString fileName = QFileInfo(item->fileName()).fileName(); | ||||||
|  |         if (fileName != widgetItem->text()) | ||||||
|  |             continue; | ||||||
|  |         found = item; | ||||||
|  |         break; | ||||||
|  |     } | ||||||
|  |     if (!found) | ||||||
|  |         qCWarning(LOG_LIB) << "Could not find item by name" << widgetItem->text(); | ||||||
|  |  | ||||||
|  |     return found; | ||||||
|  | } | ||||||
|  |  | ||||||
|  |  | ||||||
|  | void AbstractExtItemAggregator::repaintList() const | ||||||
|  | { | ||||||
|  |     ui->listWidget->clear(); | ||||||
|  |     for (auto &_item : items()) { | ||||||
|  |         QString fileName = QFileInfo(_item->fileName()).fileName(); | ||||||
|  |         auto *item = new QListWidgetItem(fileName, ui->listWidget); | ||||||
|  |         QStringList tooltip; | ||||||
|  |         tooltip.append(i18n("Name: %1", _item->name())); | ||||||
|  |         tooltip.append(i18n("Comment: %1", _item->comment())); | ||||||
|  |         tooltip.append(i18n("Identity: %1", _item->uniq())); | ||||||
|  |         item->setToolTip(tooltip.join('\n')); | ||||||
|  |         ui->listWidget->addItem(item); | ||||||
|  |     } | ||||||
|  | } | ||||||
|  |  | ||||||
|  |  | ||||||
|  | int AbstractExtItemAggregator::uniqNumber() const | ||||||
|  | { | ||||||
|  |     QList<int> tagList; | ||||||
|  |     for (auto &item : items()) | ||||||
|  |         tagList.append(item->number()); | ||||||
|  |     int number = 0; | ||||||
|  |     while (tagList.contains(number)) | ||||||
|  |         number++; | ||||||
|  |  | ||||||
|  |     return number; | ||||||
|  | } | ||||||
|  |  | ||||||
|  |  | ||||||
| QVariant AbstractExtItemAggregator::configArgs() const | QVariant AbstractExtItemAggregator::configArgs() const | ||||||
| { | { | ||||||
|     return m_configArgs; |     return m_configArgs; | ||||||
| } | } | ||||||
|  |  | ||||||
|  |  | ||||||
| void AbstractExtItemAggregator::setConfigArgs(const QVariant _configArgs) | QStringList AbstractExtItemAggregator::directories() const | ||||||
|  | { | ||||||
|  |     auto dirs = QStandardPaths::locateAll(QStandardPaths::GenericDataLocation, QString("awesomewidgets/%1").arg(type()), | ||||||
|  |                                           QStandardPaths::LocateDirectory); | ||||||
|  |  | ||||||
|  |     return dirs; | ||||||
|  | } | ||||||
|  |  | ||||||
|  |  | ||||||
|  | QString AbstractExtItemAggregator::type() const | ||||||
|  | { | ||||||
|  |     return m_type; | ||||||
|  | } | ||||||
|  |  | ||||||
|  |  | ||||||
|  | void AbstractExtItemAggregator::setConfigArgs(const QVariant &_configArgs) | ||||||
| { | { | ||||||
|     qCDebug(LOG_LIB) << "Configuration arguments" << _configArgs; |     qCDebug(LOG_LIB) << "Configuration arguments" << _configArgs; | ||||||
|  |  | ||||||
| @ -91,22 +204,20 @@ void AbstractExtItemAggregator::setConfigArgs(const QVariant _configArgs) | |||||||
| } | } | ||||||
|  |  | ||||||
|  |  | ||||||
| void AbstractExtItemAggregator::editItemActivated(QListWidgetItem *item) | void AbstractExtItemAggregator::editItemActivated(QListWidgetItem *) | ||||||
| { | { | ||||||
|     Q_UNUSED(item) |  | ||||||
|  |  | ||||||
|     return editItem(); |     return editItem(); | ||||||
| } | } | ||||||
|  |  | ||||||
|  |  | ||||||
| void AbstractExtItemAggregator::editItemButtonPressed(QAbstractButton *button) | void AbstractExtItemAggregator::editItemButtonPressed(QAbstractButton *_button) | ||||||
| { | { | ||||||
|     if (static_cast<QPushButton *>(button) == copyButton) |     if (dynamic_cast<QPushButton *>(_button) == copyButton) | ||||||
|         return copyItem(); |         return copyItem(); | ||||||
|     else if (static_cast<QPushButton *>(button) == createButton) |     else if (dynamic_cast<QPushButton *>(_button) == createButton) | ||||||
|         return createItem(); |         return doCreateItem(); | ||||||
|     else if (static_cast<QPushButton *>(button) == deleteButton) |     else if (dynamic_cast<QPushButton *>(_button) == deleteButton) | ||||||
|         return deleteItem(); |         return deleteItem(); | ||||||
|     else if (dialogButtons->buttonRole(button) == QDialogButtonBox::AcceptRole) |     else if (ui->buttonBox->buttonRole(_button) == QDialogButtonBox::AcceptRole) | ||||||
|         return editItem(); |         return editItem(); | ||||||
| } | } | ||||||
|  | |||||||
| @ -19,45 +19,80 @@ | |||||||
| #define ABSTRACTEXTITEMAGGREGATOR_H | #define ABSTRACTEXTITEMAGGREGATOR_H | ||||||
|  |  | ||||||
| #include <QDialog> | #include <QDialog> | ||||||
| #include <QDialogButtonBox> | #include <QStandardPaths> | ||||||
| #include <QListWidget> |  | ||||||
| #include <QPushButton> | #include "abstractextitem.h" | ||||||
| #include <QWidget> | #include "awdebug.h" | ||||||
|  |  | ||||||
|  |  | ||||||
| // additional class since QObject macro does not allow class templates | class QAbstractButton; | ||||||
| class AbstractExtItemAggregator : public QWidget | class QListWidgetItem; | ||||||
|  | namespace Ui | ||||||
|  | { | ||||||
|  | class AbstractExtItemAggregator; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | class AbstractExtItemAggregator : public QDialog | ||||||
| { | { | ||||||
|     Q_OBJECT |     Q_OBJECT | ||||||
|     Q_PROPERTY(QVariant configArgs READ configArgs WRITE setConfigArgs) |     Q_PROPERTY(QVariant configArgs READ configArgs WRITE setConfigArgs) | ||||||
|  |     Q_PROPERTY(QVariant type READ type) | ||||||
|  |  | ||||||
| public: | public: | ||||||
|     explicit AbstractExtItemAggregator(QWidget *parent = nullptr); |     explicit AbstractExtItemAggregator(QWidget *_parent, QString _type); | ||||||
|     virtual ~AbstractExtItemAggregator(); |     ~AbstractExtItemAggregator() override; | ||||||
|  |     // methods | ||||||
|  |     void copyItem(); | ||||||
|  |     template <class T> void createItem() | ||||||
|  |     { | ||||||
|  |         QString fileName = getName(); | ||||||
|  |         int number = uniqNumber(); | ||||||
|  |         QString dir = QString("%1/awesomewidgets/%2") | ||||||
|  |                           .arg(QStandardPaths::writableLocation(QStandardPaths::GenericDataLocation)) | ||||||
|  |                           .arg(m_type); | ||||||
|  |         if (fileName.isEmpty()) { | ||||||
|  |             qCWarning(LOG_LIB) << "Nothing to create"; | ||||||
|  |             return; | ||||||
|  |         } | ||||||
|  |         QString filePath = QString("%1/%2").arg(dir).arg(fileName); | ||||||
|  |  | ||||||
|  |         T *newItem = new T(this, filePath); | ||||||
|  |         newItem->setNumber(number); | ||||||
|  |         if (newItem->showConfiguration(configArgs()) == 1) { | ||||||
|  |             initItems(); | ||||||
|  |             repaintList(); | ||||||
|  |         } | ||||||
|  |     }; | ||||||
|  |     void deleteItem(); | ||||||
|  |     void editItem(); | ||||||
|     QString getName(); |     QString getName(); | ||||||
|  |     virtual void initItems() = 0; | ||||||
|  |     [[nodiscard]] AbstractExtItem *itemFromWidget() const; | ||||||
|  |     void repaintList() const; | ||||||
|  |     [[nodiscard]] int uniqNumber() const; | ||||||
|  |     // get methods | ||||||
|  |     [[nodiscard]] QVariant configArgs() const; | ||||||
|  |     [[nodiscard]] QStringList directories() const; | ||||||
|  |     [[nodiscard]] virtual QList<AbstractExtItem *> items() const = 0; | ||||||
|  |     [[nodiscard]] QString type() const; | ||||||
|  |     // set methods | ||||||
|  |     void setConfigArgs(const QVariant &_configArgs); | ||||||
|  |  | ||||||
|  | private slots: | ||||||
|  |     void editItemActivated(QListWidgetItem *); | ||||||
|  |     void editItemButtonPressed(QAbstractButton *_button); | ||||||
|  |  | ||||||
|  | private: | ||||||
|     // ui |     // ui | ||||||
|     QDialog *dialog = nullptr; |     Ui::AbstractExtItemAggregator *ui = nullptr; | ||||||
|     QListWidget *widgetDialog = nullptr; |  | ||||||
|     QDialogButtonBox *dialogButtons = nullptr; |  | ||||||
|     QPushButton *copyButton = nullptr; |     QPushButton *copyButton = nullptr; | ||||||
|     QPushButton *createButton = nullptr; |     QPushButton *createButton = nullptr; | ||||||
|     QPushButton *deleteButton = nullptr; |     QPushButton *deleteButton = nullptr; | ||||||
|     // get methods |     // properties | ||||||
|     QVariant configArgs() const; |  | ||||||
|     // set methods |  | ||||||
|     void setConfigArgs(const QVariant _configArgs); |  | ||||||
|  |  | ||||||
| private slots: |  | ||||||
|     void editItemActivated(QListWidgetItem *item); |  | ||||||
|     void editItemButtonPressed(QAbstractButton *button); |  | ||||||
|  |  | ||||||
| private: |  | ||||||
|     QVariant m_configArgs; |     QVariant m_configArgs; | ||||||
|     // methods |     QString m_type; | ||||||
|     virtual void copyItem() = 0; |     // ui methods | ||||||
|     virtual void createItem() = 0; |     virtual void doCreateItem() = 0; | ||||||
|     virtual void deleteItem() = 0; |  | ||||||
|     virtual void editItem() = 0; |  | ||||||
| }; | }; | ||||||
|  |  | ||||||
|  |  | ||||||
|  | |||||||
							
								
								
									
										64
									
								
								sources/awesomewidgets/abstractextitemaggregator.ui
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										64
									
								
								sources/awesomewidgets/abstractextitemaggregator.ui
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,64 @@ | |||||||
|  | <?xml version="1.0" encoding="UTF-8"?> | ||||||
|  | <ui version="4.0"> | ||||||
|  |  <class>AbstractExtItemAggregator</class> | ||||||
|  |  <widget class="QDialog" name="AbstractExtItemAggregator"> | ||||||
|  |   <property name="geometry"> | ||||||
|  |    <rect> | ||||||
|  |     <x>0</x> | ||||||
|  |     <y>0</y> | ||||||
|  |     <width>400</width> | ||||||
|  |     <height>300</height> | ||||||
|  |    </rect> | ||||||
|  |   </property> | ||||||
|  |   <layout class="QHBoxLayout" name="horizontalLayout"> | ||||||
|  |    <item> | ||||||
|  |     <widget class="QListWidget" name="listWidget"/> | ||||||
|  |    </item> | ||||||
|  |    <item> | ||||||
|  |     <widget class="QDialogButtonBox" name="buttonBox"> | ||||||
|  |      <property name="orientation"> | ||||||
|  |       <enum>Qt::Vertical</enum> | ||||||
|  |      </property> | ||||||
|  |      <property name="standardButtons"> | ||||||
|  |       <set>QDialogButtonBox::Close|QDialogButtonBox::Open</set> | ||||||
|  |      </property> | ||||||
|  |     </widget> | ||||||
|  |    </item> | ||||||
|  |   </layout> | ||||||
|  |  </widget> | ||||||
|  |  <resources/> | ||||||
|  |  <connections> | ||||||
|  |   <connection> | ||||||
|  |    <sender>buttonBox</sender> | ||||||
|  |    <signal>accepted()</signal> | ||||||
|  |    <receiver>AbstractExtItemAggregator</receiver> | ||||||
|  |    <slot>accept()</slot> | ||||||
|  |    <hints> | ||||||
|  |     <hint type="sourcelabel"> | ||||||
|  |      <x>248</x> | ||||||
|  |      <y>254</y> | ||||||
|  |     </hint> | ||||||
|  |     <hint type="destinationlabel"> | ||||||
|  |      <x>157</x> | ||||||
|  |      <y>274</y> | ||||||
|  |     </hint> | ||||||
|  |    </hints> | ||||||
|  |   </connection> | ||||||
|  |   <connection> | ||||||
|  |    <sender>buttonBox</sender> | ||||||
|  |    <signal>rejected()</signal> | ||||||
|  |    <receiver>AbstractExtItemAggregator</receiver> | ||||||
|  |    <slot>reject()</slot> | ||||||
|  |    <hints> | ||||||
|  |     <hint type="sourcelabel"> | ||||||
|  |      <x>316</x> | ||||||
|  |      <y>260</y> | ||||||
|  |     </hint> | ||||||
|  |     <hint type="destinationlabel"> | ||||||
|  |      <x>286</x> | ||||||
|  |      <y>274</y> | ||||||
|  |     </hint> | ||||||
|  |    </hints> | ||||||
|  |   </connection> | ||||||
|  |  </connections> | ||||||
|  | </ui> | ||||||
							
								
								
									
										45
									
								
								sources/awesomewidgets/abstractquotesprovider.h
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										45
									
								
								sources/awesomewidgets/abstractquotesprovider.h
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,45 @@ | |||||||
|  | /*************************************************************************** | ||||||
|  |  *   This file is part of awesome-widgets                                  * | ||||||
|  |  *                                                                         * | ||||||
|  |  *   awesome-widgets is free software: you can redistribute it and/or      * | ||||||
|  |  *   modify it under the terms of the GNU General Public License as        * | ||||||
|  |  *   published by the Free Software Foundation, either version 3 of the    * | ||||||
|  |  *   License, or (at your option) any later version.                       * | ||||||
|  |  *                                                                         * | ||||||
|  |  *   awesome-widgets is distributed in the hope that it will be useful,    * | ||||||
|  |  *   but WITHOUT ANY WARRANTY; without even the implied warranty of        * | ||||||
|  |  *   MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the         * | ||||||
|  |  *   GNU General Public License for more details.                          * | ||||||
|  |  *                                                                         * | ||||||
|  |  *   You should have received a copy of the GNU General Public License     * | ||||||
|  |  *   along with awesome-widgets. If not, see http://www.gnu.org/licenses/  * | ||||||
|  |  ***************************************************************************/ | ||||||
|  |  | ||||||
|  | #ifndef ABSTRACTQUOTESPROVIDER_H | ||||||
|  | #define ABSTRACTQUOTESPROVIDER_H | ||||||
|  |  | ||||||
|  | #include <QObject> | ||||||
|  | #include <QUrl> | ||||||
|  |  | ||||||
|  | #include "abstractextitem.h" | ||||||
|  |  | ||||||
|  |  | ||||||
|  | class AbstractQuotesProvider : public QObject | ||||||
|  | { | ||||||
|  |     Q_OBJECT | ||||||
|  |  | ||||||
|  | public: | ||||||
|  |     explicit AbstractQuotesProvider(QObject *_parent) | ||||||
|  |         : QObject(_parent){}; | ||||||
|  |     ~AbstractQuotesProvider() override = default; | ||||||
|  |     virtual void initUrl(const QString &_asset) = 0; | ||||||
|  |     [[nodiscard]] virtual QVariantHash parse(const QByteArray &_source, const QVariantHash &_oldValues) const = 0; | ||||||
|  |     [[nodiscard]] QString tag(const QString &_type) const | ||||||
|  |     { | ||||||
|  |         return dynamic_cast<AbstractExtItem *>(parent())->tag(_type); | ||||||
|  |     }; | ||||||
|  |     [[nodiscard]] virtual QUrl url() const = 0; | ||||||
|  | }; | ||||||
|  |  | ||||||
|  |  | ||||||
|  | #endif /* ABSTRACTQUOTESPROVIDER_H */ | ||||||
							
								
								
									
										45
									
								
								sources/awesomewidgets/abstractweatherprovider.h
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										45
									
								
								sources/awesomewidgets/abstractweatherprovider.h
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,45 @@ | |||||||
|  | /*************************************************************************** | ||||||
|  |  *   This file is part of awesome-widgets                                  * | ||||||
|  |  *                                                                         * | ||||||
|  |  *   awesome-widgets is free software: you can redistribute it and/or      * | ||||||
|  |  *   modify it under the terms of the GNU General Public License as        * | ||||||
|  |  *   published by the Free Software Foundation, either version 3 of the    * | ||||||
|  |  *   License, or (at your option) any later version.                       * | ||||||
|  |  *                                                                         * | ||||||
|  |  *   awesome-widgets is distributed in the hope that it will be useful,    * | ||||||
|  |  *   but WITHOUT ANY WARRANTY; without even the implied warranty of        * | ||||||
|  |  *   MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the         * | ||||||
|  |  *   GNU General Public License for more details.                          * | ||||||
|  |  *                                                                         * | ||||||
|  |  *   You should have received a copy of the GNU General Public License     * | ||||||
|  |  *   along with awesome-widgets. If not, see http://www.gnu.org/licenses/  * | ||||||
|  |  ***************************************************************************/ | ||||||
|  |  | ||||||
|  | #ifndef ABSTRACTWEATHERPROVIDER_H | ||||||
|  | #define ABSTRACTWEATHERPROVIDER_H | ||||||
|  |  | ||||||
|  | #include <QObject> | ||||||
|  | #include <QUrl> | ||||||
|  |  | ||||||
|  | #include "abstractextitem.h" | ||||||
|  |  | ||||||
|  |  | ||||||
|  | class AbstractWeatherProvider : public QObject | ||||||
|  | { | ||||||
|  |     Q_OBJECT | ||||||
|  |  | ||||||
|  | public: | ||||||
|  |     explicit AbstractWeatherProvider(QObject *_parent) | ||||||
|  |         : QObject(_parent){}; | ||||||
|  |     ~AbstractWeatherProvider() override = default; | ||||||
|  |     virtual void initUrl(const QString &_city, const QString &_country, int _ts) = 0; | ||||||
|  |     [[nodiscard]] virtual QVariantHash parse(const QVariantMap &_json) const = 0; | ||||||
|  |     [[nodiscard]] QString tag(const QString &_type) const | ||||||
|  |     { | ||||||
|  |         return dynamic_cast<AbstractExtItem *>(parent())->tag(_type); | ||||||
|  |     }; | ||||||
|  |     [[nodiscard]] virtual QUrl url() const = 0; | ||||||
|  | }; | ||||||
|  |  | ||||||
|  |  | ||||||
|  | #endif /* ABSTRACTWEATHERPROVIDER_H */ | ||||||
							
								
								
									
										142
									
								
								sources/awesomewidgets/awabstractformatter.cpp
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										142
									
								
								sources/awesomewidgets/awabstractformatter.cpp
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,142 @@ | |||||||
|  | /*************************************************************************** | ||||||
|  |  *   This file is part of awesome-widgets                                  * | ||||||
|  |  *                                                                         * | ||||||
|  |  *   awesome-widgets is free software: you can redistribute it and/or      * | ||||||
|  |  *   modify it under the terms of the GNU General Public License as        * | ||||||
|  |  *   published by the Free Software Foundation, either version 3 of the    * | ||||||
|  |  *   License, or (at your option) any later version.                       * | ||||||
|  |  *                                                                         * | ||||||
|  |  *   awesome-widgets is distributed in the hope that it will be useful,    * | ||||||
|  |  *   but WITHOUT ANY WARRANTY; without even the implied warranty of        * | ||||||
|  |  *   MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the         * | ||||||
|  |  *   GNU General Public License for more details.                          * | ||||||
|  |  *                                                                         * | ||||||
|  |  *   You should have received a copy of the GNU General Public License     * | ||||||
|  |  *   along with awesome-widgets. If not, see http://www.gnu.org/licenses/  * | ||||||
|  |  ***************************************************************************/ | ||||||
|  |  | ||||||
|  | #include "awabstractformatter.h" | ||||||
|  |  | ||||||
|  | #include <QSettings> | ||||||
|  |  | ||||||
|  | #include "awdebug.h" | ||||||
|  |  | ||||||
|  |  | ||||||
|  | AWAbstractFormatter::AWAbstractFormatter(QWidget *_parent, const QString &_filePath) | ||||||
|  |     : AbstractExtItem(_parent, _filePath) | ||||||
|  | { | ||||||
|  |     qCDebug(LOG_LIB) << __PRETTY_FUNCTION__; | ||||||
|  | } | ||||||
|  |  | ||||||
|  |  | ||||||
|  | AWAbstractFormatter::~AWAbstractFormatter() | ||||||
|  | { | ||||||
|  |     qCDebug(LOG_LIB) << __PRETTY_FUNCTION__; | ||||||
|  | } | ||||||
|  |  | ||||||
|  |  | ||||||
|  | void AWAbstractFormatter::copyDefaults(AbstractExtItem *_other) const | ||||||
|  | { | ||||||
|  |     AbstractExtItem::copyDefaults(_other); | ||||||
|  |  | ||||||
|  |     dynamic_cast<AWAbstractFormatter *>(_other)->setType(type()); | ||||||
|  | } | ||||||
|  |  | ||||||
|  |  | ||||||
|  | QString AWAbstractFormatter::uniq() const | ||||||
|  | { | ||||||
|  |     return QString("%1(%2)").arg(name()).arg(strType()); | ||||||
|  | } | ||||||
|  |  | ||||||
|  |  | ||||||
|  | QString AWAbstractFormatter::strType() const | ||||||
|  | { | ||||||
|  |     QString value; | ||||||
|  |     switch (m_type) { | ||||||
|  |     case FormatterClass::DateTime: | ||||||
|  |         value = "DateTime"; | ||||||
|  |         break; | ||||||
|  |     case FormatterClass::Float: | ||||||
|  |         value = "Float"; | ||||||
|  |         break; | ||||||
|  |     case FormatterClass::List: | ||||||
|  |         value = "List"; | ||||||
|  |         break; | ||||||
|  |     case FormatterClass::Script: | ||||||
|  |         value = "Script"; | ||||||
|  |         break; | ||||||
|  |     case FormatterClass::String: | ||||||
|  |         value = "String"; | ||||||
|  |         break; | ||||||
|  |     case FormatterClass::Json: | ||||||
|  |         value = "Json"; | ||||||
|  |         break; | ||||||
|  |     case FormatterClass::NoFormat: | ||||||
|  |         value = "NoFormat"; | ||||||
|  |         break; | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     return value; | ||||||
|  | } | ||||||
|  |  | ||||||
|  |  | ||||||
|  | AWAbstractFormatter::FormatterClass AWAbstractFormatter::type() const | ||||||
|  | { | ||||||
|  |     return m_type; | ||||||
|  | } | ||||||
|  |  | ||||||
|  |  | ||||||
|  | void AWAbstractFormatter::setStrType(const QString &_type) | ||||||
|  | { | ||||||
|  |     qCDebug(LOG_LIB) << "Type" << _type; | ||||||
|  |  | ||||||
|  |     if (_type == "DateTime") | ||||||
|  |         m_type = FormatterClass::DateTime; | ||||||
|  |     else if (_type == "Float") | ||||||
|  |         m_type = FormatterClass::Float; | ||||||
|  |     else if (_type == "List") | ||||||
|  |         m_type = FormatterClass::List; | ||||||
|  |     else if (_type == "Script") | ||||||
|  |         m_type = FormatterClass::Script; | ||||||
|  |     else if (_type == "String") | ||||||
|  |         m_type = FormatterClass::String; | ||||||
|  |     else if (_type == "Json") | ||||||
|  |         m_type = FormatterClass::Json; | ||||||
|  |     else | ||||||
|  |         m_type = FormatterClass::NoFormat; | ||||||
|  | } | ||||||
|  |  | ||||||
|  |  | ||||||
|  | void AWAbstractFormatter::setType(const AWAbstractFormatter::FormatterClass _type) | ||||||
|  | { | ||||||
|  |     qCDebug(LOG_LIB) << "Type" << static_cast<int>(_type); | ||||||
|  |  | ||||||
|  |     m_type = _type; | ||||||
|  | } | ||||||
|  |  | ||||||
|  |  | ||||||
|  | void AWAbstractFormatter::readConfiguration() | ||||||
|  | { | ||||||
|  |     AbstractExtItem::readConfiguration(); | ||||||
|  |  | ||||||
|  |     QSettings settings(fileName(), QSettings::IniFormat); | ||||||
|  |  | ||||||
|  |     settings.beginGroup("Desktop Entry"); | ||||||
|  |     setStrType(settings.value("X-AW-Type", strType()).toString()); | ||||||
|  |     settings.endGroup(); | ||||||
|  | } | ||||||
|  |  | ||||||
|  |  | ||||||
|  | void AWAbstractFormatter::writeConfiguration() const | ||||||
|  | { | ||||||
|  |     AbstractExtItem::writeConfiguration(); | ||||||
|  |  | ||||||
|  |     QSettings settings(writtableConfig(), QSettings::IniFormat); | ||||||
|  |     qCInfo(LOG_LIB) << "Configuration file" << settings.fileName(); | ||||||
|  |  | ||||||
|  |     settings.beginGroup("Desktop Entry"); | ||||||
|  |     settings.setValue("X-AW-Type", strType()); | ||||||
|  |     settings.endGroup(); | ||||||
|  |  | ||||||
|  |     settings.sync(); | ||||||
|  | } | ||||||
Some files were not shown because too many files have changed in this diff Show More
		Reference in New Issue
	
	Block a user
	