diff --git a/.gitignore b/.gitignore
index 68bc17f..2dc53ca 100644
--- a/.gitignore
+++ b/.gitignore
@@ -157,4 +157,4 @@ cython_debug/
# be found at https://github.com/github/gitignore/blob/main/Global/JetBrains.gitignore
# and can be added to the global gitignore or merged into this file. For a more nuclear
# option (not recommended) you can uncomment the following to ignore the entire idea folder.
-#.idea/
+.idea/
diff --git a/.python-version b/.python-version
new file mode 100644
index 0000000..171a6a9
--- /dev/null
+++ b/.python-version
@@ -0,0 +1 @@
+3.12.1
diff --git a/Pipfile b/Pipfile
new file mode 100644
index 0000000..44cd01d
--- /dev/null
+++ b/Pipfile
@@ -0,0 +1,15 @@
+[[source]]
+url = "https://pypi.org/simple"
+verify_ssl = true
+name = "pypi"
+
+[packages]
+pyside6 = "==6.6.1"
+requests = "==2.31.0"
+wmi = "==1.5.1"
+pyinstaller = "==6.3.0"
+
+[dev-packages]
+
+[requires]
+python_version = "3.12"
diff --git a/Pipfile.lock b/Pipfile.lock
new file mode 100644
index 0000000..4d22bf4
--- /dev/null
+++ b/Pipfile.lock
@@ -0,0 +1,284 @@
+{
+ "_meta": {
+ "hash": {
+ "sha256": "c9e6c224ff1fb7073100c794e46ce72371b7c482473c9704fc5709aefe9f1f33"
+ },
+ "pipfile-spec": 6,
+ "requires": {
+ "python_version": "3.12"
+ },
+ "sources": [
+ {
+ "name": "pypi",
+ "url": "https://pypi.org/simple",
+ "verify_ssl": true
+ }
+ ]
+ },
+ "default": {
+ "altgraph": {
+ "hashes": [
+ "sha256:1b5afbb98f6c4dcadb2e2ae6ab9fa994bbb8c1d75f4fa96d340f9437ae454406",
+ "sha256:642743b4750de17e655e6711601b077bc6598dbfa3ba5fa2b2a35ce12b508dff"
+ ],
+ "version": "==0.17.4"
+ },
+ "certifi": {
+ "hashes": [
+ "sha256:0569859f95fc761b18b45ef421b1290a0f65f147e92a1e5eb3e635f9a5e4e66f",
+ "sha256:dc383c07b76109f368f6106eee2b593b04a011ea4d55f652c6ca24a754d1cdd1"
+ ],
+ "markers": "python_version >= '3.6'",
+ "version": "==2024.2.2"
+ },
+ "charset-normalizer": {
+ "hashes": [
+ "sha256:06435b539f889b1f6f4ac1758871aae42dc3a8c0e24ac9e60c2384973ad73027",
+ "sha256:06a81e93cd441c56a9b65d8e1d043daeb97a3d0856d177d5c90ba85acb3db087",
+ "sha256:0a55554a2fa0d408816b3b5cedf0045f4b8e1a6065aec45849de2d6f3f8e9786",
+ "sha256:0b2b64d2bb6d3fb9112bafa732def486049e63de9618b5843bcdd081d8144cd8",
+ "sha256:10955842570876604d404661fbccbc9c7e684caf432c09c715ec38fbae45ae09",
+ "sha256:122c7fa62b130ed55f8f285bfd56d5f4b4a5b503609d181f9ad85e55c89f4185",
+ "sha256:1ceae2f17a9c33cb48e3263960dc5fc8005351ee19db217e9b1bb15d28c02574",
+ "sha256:1d3193f4a680c64b4b6a9115943538edb896edc190f0b222e73761716519268e",
+ "sha256:1f79682fbe303db92bc2b1136016a38a42e835d932bab5b3b1bfcfbf0640e519",
+ "sha256:2127566c664442652f024c837091890cb1942c30937add288223dc895793f898",
+ "sha256:22afcb9f253dac0696b5a4be4a1c0f8762f8239e21b99680099abd9b2b1b2269",
+ "sha256:25baf083bf6f6b341f4121c2f3c548875ee6f5339300e08be3f2b2ba1721cdd3",
+ "sha256:2e81c7b9c8979ce92ed306c249d46894776a909505d8f5a4ba55b14206e3222f",
+ "sha256:3287761bc4ee9e33561a7e058c72ac0938c4f57fe49a09eae428fd88aafe7bb6",
+ "sha256:34d1c8da1e78d2e001f363791c98a272bb734000fcef47a491c1e3b0505657a8",
+ "sha256:37e55c8e51c236f95b033f6fb391d7d7970ba5fe7ff453dad675e88cf303377a",
+ "sha256:3d47fa203a7bd9c5b6cee4736ee84ca03b8ef23193c0d1ca99b5089f72645c73",
+ "sha256:3e4d1f6587322d2788836a99c69062fbb091331ec940e02d12d179c1d53e25fc",
+ "sha256:42cb296636fcc8b0644486d15c12376cb9fa75443e00fb25de0b8602e64c1714",
+ "sha256:45485e01ff4d3630ec0d9617310448a8702f70e9c01906b0d0118bdf9d124cf2",
+ "sha256:4a78b2b446bd7c934f5dcedc588903fb2f5eec172f3d29e52a9096a43722adfc",
+ "sha256:4ab2fe47fae9e0f9dee8c04187ce5d09f48eabe611be8259444906793ab7cbce",
+ "sha256:4d0d1650369165a14e14e1e47b372cfcb31d6ab44e6e33cb2d4e57265290044d",
+ "sha256:549a3a73da901d5bc3ce8d24e0600d1fa85524c10287f6004fbab87672bf3e1e",
+ "sha256:55086ee1064215781fff39a1af09518bc9255b50d6333f2e4c74ca09fac6a8f6",
+ "sha256:572c3763a264ba47b3cf708a44ce965d98555f618ca42c926a9c1616d8f34269",
+ "sha256:573f6eac48f4769d667c4442081b1794f52919e7edada77495aaed9236d13a96",
+ "sha256:5b4c145409bef602a690e7cfad0a15a55c13320ff7a3ad7ca59c13bb8ba4d45d",
+ "sha256:6463effa3186ea09411d50efc7d85360b38d5f09b870c48e4600f63af490e56a",
+ "sha256:65f6f63034100ead094b8744b3b97965785388f308a64cf8d7c34f2f2e5be0c4",
+ "sha256:663946639d296df6a2bb2aa51b60a2454ca1cb29835324c640dafb5ff2131a77",
+ "sha256:6897af51655e3691ff853668779c7bad41579facacf5fd7253b0133308cf000d",
+ "sha256:68d1f8a9e9e37c1223b656399be5d6b448dea850bed7d0f87a8311f1ff3dabb0",
+ "sha256:6ac7ffc7ad6d040517be39eb591cac5ff87416c2537df6ba3cba3bae290c0fed",
+ "sha256:6b3251890fff30ee142c44144871185dbe13b11bab478a88887a639655be1068",
+ "sha256:6c4caeef8fa63d06bd437cd4bdcf3ffefe6738fb1b25951440d80dc7df8c03ac",
+ "sha256:6ef1d82a3af9d3eecdba2321dc1b3c238245d890843e040e41e470ffa64c3e25",
+ "sha256:753f10e867343b4511128c6ed8c82f7bec3bd026875576dfd88483c5c73b2fd8",
+ "sha256:7cd13a2e3ddeed6913a65e66e94b51d80a041145a026c27e6bb76c31a853c6ab",
+ "sha256:7ed9e526742851e8d5cc9e6cf41427dfc6068d4f5a3bb03659444b4cabf6bc26",
+ "sha256:7f04c839ed0b6b98b1a7501a002144b76c18fb1c1850c8b98d458ac269e26ed2",
+ "sha256:802fe99cca7457642125a8a88a084cef28ff0cf9407060f7b93dca5aa25480db",
+ "sha256:80402cd6ee291dcb72644d6eac93785fe2c8b9cb30893c1af5b8fdd753b9d40f",
+ "sha256:8465322196c8b4d7ab6d1e049e4c5cb460d0394da4a27d23cc242fbf0034b6b5",
+ "sha256:86216b5cee4b06df986d214f664305142d9c76df9b6512be2738aa72a2048f99",
+ "sha256:87d1351268731db79e0f8e745d92493ee2841c974128ef629dc518b937d9194c",
+ "sha256:8bdb58ff7ba23002a4c5808d608e4e6c687175724f54a5dade5fa8c67b604e4d",
+ "sha256:8c622a5fe39a48f78944a87d4fb8a53ee07344641b0562c540d840748571b811",
+ "sha256:8d756e44e94489e49571086ef83b2bb8ce311e730092d2c34ca8f7d925cb20aa",
+ "sha256:8f4a014bc36d3c57402e2977dada34f9c12300af536839dc38c0beab8878f38a",
+ "sha256:9063e24fdb1e498ab71cb7419e24622516c4a04476b17a2dab57e8baa30d6e03",
+ "sha256:90d558489962fd4918143277a773316e56c72da56ec7aa3dc3dbbe20fdfed15b",
+ "sha256:923c0c831b7cfcb071580d3f46c4baf50f174be571576556269530f4bbd79d04",
+ "sha256:95f2a5796329323b8f0512e09dbb7a1860c46a39da62ecb2324f116fa8fdc85c",
+ "sha256:96b02a3dc4381e5494fad39be677abcb5e6634bf7b4fa83a6dd3112607547001",
+ "sha256:9f96df6923e21816da7e0ad3fd47dd8f94b2a5ce594e00677c0013018b813458",
+ "sha256:a10af20b82360ab00827f916a6058451b723b4e65030c5a18577c8b2de5b3389",
+ "sha256:a50aebfa173e157099939b17f18600f72f84eed3049e743b68ad15bd69b6bf99",
+ "sha256:a981a536974bbc7a512cf44ed14938cf01030a99e9b3a06dd59578882f06f985",
+ "sha256:a9a8e9031d613fd2009c182b69c7b2c1ef8239a0efb1df3f7c8da66d5dd3d537",
+ "sha256:ae5f4161f18c61806f411a13b0310bea87f987c7d2ecdbdaad0e94eb2e404238",
+ "sha256:aed38f6e4fb3f5d6bf81bfa990a07806be9d83cf7bacef998ab1a9bd660a581f",
+ "sha256:b01b88d45a6fcb69667cd6d2f7a9aeb4bf53760d7fc536bf679ec94fe9f3ff3d",
+ "sha256:b261ccdec7821281dade748d088bb6e9b69e6d15b30652b74cbbac25e280b796",
+ "sha256:b2b0a0c0517616b6869869f8c581d4eb2dd83a4d79e0ebcb7d373ef9956aeb0a",
+ "sha256:b4a23f61ce87adf89be746c8a8974fe1c823c891d8f86eb218bb957c924bb143",
+ "sha256:bd8f7df7d12c2db9fab40bdd87a7c09b1530128315d047a086fa3ae3435cb3a8",
+ "sha256:beb58fe5cdb101e3a055192ac291b7a21e3b7ef4f67fa1d74e331a7f2124341c",
+ "sha256:c002b4ffc0be611f0d9da932eb0f704fe2602a9a949d1f738e4c34c75b0863d5",
+ "sha256:c083af607d2515612056a31f0a8d9e0fcb5876b7bfc0abad3ecd275bc4ebc2d5",
+ "sha256:c180f51afb394e165eafe4ac2936a14bee3eb10debc9d9e4db8958fe36afe711",
+ "sha256:c235ebd9baae02f1b77bcea61bce332cb4331dc3617d254df3323aa01ab47bd4",
+ "sha256:cd70574b12bb8a4d2aaa0094515df2463cb429d8536cfb6c7ce983246983e5a6",
+ "sha256:d0eccceffcb53201b5bfebb52600a5fb483a20b61da9dbc885f8b103cbe7598c",
+ "sha256:d965bba47ddeec8cd560687584e88cf699fd28f192ceb452d1d7ee807c5597b7",
+ "sha256:db364eca23f876da6f9e16c9da0df51aa4f104a972735574842618b8c6d999d4",
+ "sha256:ddbb2551d7e0102e7252db79ba445cdab71b26640817ab1e3e3648dad515003b",
+ "sha256:deb6be0ac38ece9ba87dea880e438f25ca3eddfac8b002a2ec3d9183a454e8ae",
+ "sha256:e06ed3eb3218bc64786f7db41917d4e686cc4856944f53d5bdf83a6884432e12",
+ "sha256:e27ad930a842b4c5eb8ac0016b0a54f5aebbe679340c26101df33424142c143c",
+ "sha256:e537484df0d8f426ce2afb2d0f8e1c3d0b114b83f8850e5f2fbea0e797bd82ae",
+ "sha256:eb00ed941194665c332bf8e078baf037d6c35d7c4f3102ea2d4f16ca94a26dc8",
+ "sha256:eb6904c354526e758fda7167b33005998fb68c46fbc10e013ca97f21ca5c8887",
+ "sha256:eb8821e09e916165e160797a6c17edda0679379a4be5c716c260e836e122f54b",
+ "sha256:efcb3f6676480691518c177e3b465bcddf57cea040302f9f4e6e191af91174d4",
+ "sha256:f27273b60488abe721a075bcca6d7f3964f9f6f067c8c4c605743023d7d3944f",
+ "sha256:f30c3cb33b24454a82faecaf01b19c18562b1e89558fb6c56de4d9118a032fd5",
+ "sha256:fb69256e180cb6c8a894fee62b3afebae785babc1ee98b81cdf68bbca1987f33",
+ "sha256:fd1abc0d89e30cc4e02e4064dc67fcc51bd941eb395c502aac3ec19fab46b519",
+ "sha256:ff8fa367d09b717b2a17a052544193ad76cd49979c805768879cb63d9ca50561"
+ ],
+ "markers": "python_full_version >= '3.7.0'",
+ "version": "==3.3.2"
+ },
+ "idna": {
+ "hashes": [
+ "sha256:9ecdbbd083b06798ae1e86adcbfe8ab1479cf864e4ee30fe4e46a003d12491ca",
+ "sha256:c05567e9c24a6b9faaa835c4821bad0590fbb9d5779e7caa6e1cc4978e7eb24f"
+ ],
+ "markers": "python_version >= '3.5'",
+ "version": "==3.6"
+ },
+ "packaging": {
+ "hashes": [
+ "sha256:048fb0e9405036518eaaf48a55953c750c11e1a1b68e0dd1a9d62ed0c092cfc5",
+ "sha256:8c491190033a9af7e1d931d0b5dacc2ef47509b34dd0de67ed209b5203fc88c7"
+ ],
+ "markers": "python_version >= '3.7'",
+ "version": "==23.2"
+ },
+ "pefile": {
+ "hashes": [
+ "sha256:82e6114004b3d6911c77c3953e3838654b04511b8b66e8583db70c65998017dc",
+ "sha256:da185cd2af68c08a6cd4481f7325ed600a88f6a813bad9dea07ab3ef73d8d8d6"
+ ],
+ "markers": "sys_platform == 'win32'",
+ "version": "==2023.2.7"
+ },
+ "pyinstaller": {
+ "hashes": [
+ "sha256:0597fb04337695e5cc5250253e0655530bf14f264b7a5b7d219cc65f6889c4bd",
+ "sha256:156b32ba943e0090bcc68e40ae1cb68fd92b7f1ab6fe0bdf8faf3d3cfc4e12dd",
+ "sha256:1eadbd1fae84e2e6c678d8b4ed6a232ec5c8fe3a839aea5a3071c4c0282f98cc",
+ "sha256:41c937fe8f07ae02009b3b5a96ac3eb0800a4f8a97af142d4100060fe2135bb9",
+ "sha256:75a6f2a6f835a2e6e0899d10e60c10caf5defd25aced38b1dd48fbbabc89de07",
+ "sha256:886b3b995b674905a20ad5b720b47cc395897d7b391117831027a4c8c5d67a58",
+ "sha256:914d4c96cc99472e37ac552fdd82fbbe09e67bb592d0717fcffaa99ea74273df",
+ "sha256:96c37a1ee5b2fd5bb25c098ef510661d6d17b6515d0b86d8fc93727dd2475ba3",
+ "sha256:abe91106a3bbccc3f3a27af4325676ecdb6f46cb842ac663625002a870fc503b",
+ "sha256:b721d793a33b6d9946c7dd95d3ea7589c0424b51cf1b9fe580f03c544f1336b2",
+ "sha256:de25beb176f73a944758553caacec46cc665bf3910ad8a174706d79cf6e95340",
+ "sha256:e436fcc0ea87c3f132baac916d508c24c84a8f6d8a06c3154fbc753f169b76c7"
+ ],
+ "index": "pypi",
+ "markers": "python_version < '3.13' and python_version >= '3.8'",
+ "version": "==6.3.0"
+ },
+ "pyinstaller-hooks-contrib": {
+ "hashes": [
+ "sha256:469b5690df53223e2e8abffb2e44d6ee596e7d79d4b1eed9465123b67439875a",
+ "sha256:a7118c1a5c9788595e5c43ad058a7a5b7b6d59e1eceb42362f6ec1f0b61986b0"
+ ],
+ "markers": "python_version >= '3.7'",
+ "version": "==2024.0"
+ },
+ "pyside6": {
+ "hashes": [
+ "sha256:0a67587c088cb80e90d4ce3023b02466ea858c93a6dc9c4e062b13314e03d464",
+ "sha256:3593d605175e83e6952cf3b428ecc9c146af97effb36de921ecf3da2752de082",
+ "sha256:3c348948fe3957b18164c9c7b8942fe065bdb39648b326f212bc114326679fa9",
+ "sha256:ed3822150f0d7a06b68bf4ceebe287515b5e8309bb256e9b49ae405afd062b18"
+ ],
+ "index": "pypi",
+ "markers": "python_version < '3.13' and python_version >= '3.8'",
+ "version": "==6.6.1"
+ },
+ "pyside6-addons": {
+ "hashes": [
+ "sha256:5a63a8a943724ce5acd2df72e5ab04982b6906963278cbabb216656b9a26ee18",
+ "sha256:7cb7af1b050c40f7ac891b0e61c758c1923863173932f5b92dc47bdfb4158b42",
+ "sha256:a0982da4033319667f9df5ed6fa8eff300a88216aec103a1fff6751a172b19a0",
+ "sha256:a223575c81e9a13173136c044c3447e25f6d656b462b4d71fc3c6bd9c935a709"
+ ],
+ "markers": "python_version < '3.13' and python_version >= '3.8'",
+ "version": "==6.6.1"
+ },
+ "pyside6-essentials": {
+ "hashes": [
+ "sha256:0c8917b15236956957178a8c9854641b12b11dad79ba0caf26147119164c30cf",
+ "sha256:13da926e9e9ee3e26e3f66883a9d5e43726ddee70cdabddca02a07aa1ccf9484",
+ "sha256:a383c3d60298392cfb621ec1a0cf24b4799321e6c5bbafc021d4cc8076ea1315",
+ "sha256:c7185616083eab6f42eaed598d97d49fac4f60ae2e7415194140d54f58c2b42c"
+ ],
+ "markers": "python_version < '3.13' and python_version >= '3.8'",
+ "version": "==6.6.1"
+ },
+ "pywin32": {
+ "hashes": [
+ "sha256:06d3420a5155ba65f0b72f2699b5bacf3109f36acbe8923765c22938a69dfc8d",
+ "sha256:1c73ea9a0d2283d889001998059f5eaaba3b6238f767c9cf2833b13e6a685f65",
+ "sha256:37257794c1ad39ee9be652da0462dc2e394c8159dfd913a8a4e8eb6fd346da0e",
+ "sha256:383229d515657f4e3ed1343da8be101000562bf514591ff383ae940cad65458b",
+ "sha256:39b61c15272833b5c329a2989999dcae836b1eed650252ab1b7bfbe1d59f30f4",
+ "sha256:5821ec52f6d321aa59e2db7e0a35b997de60c201943557d108af9d4ae1ec7040",
+ "sha256:70dba0c913d19f942a2db25217d9a1b726c278f483a919f1abfed79c9cf64d3a",
+ "sha256:72c5f621542d7bdd4fdb716227be0dd3f8565c11b280be6315b06ace35487d36",
+ "sha256:84f4471dbca1887ea3803d8848a1616429ac94a4a8d05f4bc9c5dcfd42ca99c8",
+ "sha256:a7639f51c184c0272e93f244eb24dafca9b1855707d94c192d4a0b4c01e1100e",
+ "sha256:e25fd5b485b55ac9c057f67d94bc203f3f6595078d1fb3b458c9c28b7153a802",
+ "sha256:e4c092e2589b5cf0d365849e73e02c391c1349958c5ac3e9d5ccb9a28e017b3a",
+ "sha256:e65028133d15b64d2ed8f06dd9fbc268352478d4f9289e69c190ecd6818b6407",
+ "sha256:e8ac1ae3601bee6ca9f7cb4b5363bf1c0badb935ef243c4733ff9a393b1690c0"
+ ],
+ "version": "==306"
+ },
+ "pywin32-ctypes": {
+ "hashes": [
+ "sha256:3426e063bdd5fd4df74a14fa3cf80a0b42845a87e1d1e81f6549f9daec593a60",
+ "sha256:bf490a1a709baf35d688fe0ecf980ed4de11d2b3e37b51e5442587a75d9957e7"
+ ],
+ "markers": "sys_platform == 'win32'",
+ "version": "==0.2.2"
+ },
+ "requests": {
+ "hashes": [
+ "sha256:58cd2187c01e70e6e26505bca751777aa9f2ee0b7f4300988b709f44e013003f",
+ "sha256:942c5a758f98d790eaed1a29cb6eefc7ffb0d1cf7af05c3d2791656dbd6ad1e1"
+ ],
+ "index": "pypi",
+ "markers": "python_version >= '3.7'",
+ "version": "==2.31.0"
+ },
+ "setuptools": {
+ "hashes": [
+ "sha256:385eb4edd9c9d5c17540511303e39a147ce2fc04bc55289c322b9e5904fe2c05",
+ "sha256:be1af57fc409f93647f2e8e4573a142ed38724b8cdd389706a867bb4efcf1e78"
+ ],
+ "markers": "python_version >= '3.8'",
+ "version": "==69.0.3"
+ },
+ "shiboken6": {
+ "hashes": [
+ "sha256:072c35c4fe46ec13b364d9dc47b055bb2277ee3aeaab18c23650280ec362f62a",
+ "sha256:a605960e72af5eef915991cee7eef4cc72f5cabe63b9ae1a955ceb3d3b0a00b9",
+ "sha256:d756fd1fa945b787e8eef142f2eb571da0b4c4dc2f2eec1a7c12a474a2cf84e4",
+ "sha256:fb102e4bc210006f0cdd0ce38e1aaaaf792bd871f02a2b3f01d07922c5cf4c59"
+ ],
+ "markers": "python_version < '3.13' and python_version >= '3.8'",
+ "version": "==6.6.1"
+ },
+ "urllib3": {
+ "hashes": [
+ "sha256:051d961ad0c62a94e50ecf1af379c3aba230c66c710493493560c0c223c49f20",
+ "sha256:ce3711610ddce217e6d113a2732fafad960a03fd0318c91faa79481e35c11224"
+ ],
+ "markers": "python_version >= '3.8'",
+ "version": "==2.2.0"
+ },
+ "wmi": {
+ "hashes": [
+ "sha256:1d6b085e5c445141c475476000b661f60fff1aaa19f76bf82b7abb92e0ff4942",
+ "sha256:b6a6be5711b1b6c8d55bda7a8befd75c48c12b770b9d227d31c1737dbf0d40a6"
+ ],
+ "index": "pypi",
+ "version": "==1.5.1"
+ }
+ },
+ "develop": {}
+}
diff --git a/README.md b/README.md
new file mode 100644
index 0000000..475148a
--- /dev/null
+++ b/README.md
@@ -0,0 +1,10 @@
+# DailyCheck 日常检查工具
+
+适用平台:Windows
+
+### 版本日志:
+
+#### v1.0.0
+
+初始版本
+
diff --git a/da_ext_settings.py b/da_ext_settings.py
new file mode 100644
index 0000000..00a99b3
--- /dev/null
+++ b/da_ext_settings.py
@@ -0,0 +1,159 @@
+# coding: utf8
+from PySide6 import QtWidgets, QtCore
+
+
+class PushButtonWithId(QtWidgets.QPushButton):
+
+ clicked_with_id = QtCore.Signal(str)
+
+ def __init__(self, ids: str, parent: QtWidgets.QWidget = None, title: str = ""):
+ super().__init__(title, parent)
+ self.ids = ids
+ self.clicked.connect(self.on_self_clicked)
+
+ def on_self_clicked(self):
+ self.clicked_with_id.emit(self.ids)
+
+
+class DaExtSettings(QtWidgets.QDialog):
+
+ def __init__(self, parent: QtWidgets.QWidget = None):
+ super().__init__(parent)
+ self.setWindowTitle("设置")
+
+ self.vly_m = QtWidgets.QVBoxLayout()
+ self.setLayout(self.vly_m)
+
+ self.gbx_exec = QtWidgets.QGroupBox("执行文件路径", self)
+ self.vly_m.addWidget(self.gbx_exec)
+
+ self.gly_gbx_exec = QtWidgets.QGridLayout()
+ self.gbx_exec.setLayout(self.gly_gbx_exec)
+
+ self.lb_exec_chrome = QtWidgets.QLabel("Chrome", self)
+ self.lb_exec_edge = QtWidgets.QLabel("Edge", self)
+ self.lb_exec_brave = QtWidgets.QLabel("Brave", self)
+ self.lne_exec_chrome = QtWidgets.QLineEdit(self)
+ self.lne_exec_edge = QtWidgets.QLineEdit(self)
+ self.lne_exec_brave = QtWidgets.QLineEdit(self)
+ self.pbn_exec_chrome = PushButtonWithId("ChromeExec", self, "选择")
+ self.pbn_exec_edge = PushButtonWithId("EdgeExec", self, "选择")
+ self.pbn_exec_brave = PushButtonWithId("BraveExec", self, "选择")
+
+ self.gly_gbx_exec.addWidget(self.lb_exec_chrome, 0, 0)
+ self.gly_gbx_exec.addWidget(self.lb_exec_edge, 1, 0)
+ self.gly_gbx_exec.addWidget(self.lb_exec_brave, 2, 0)
+ self.gly_gbx_exec.addWidget(self.lne_exec_chrome, 0, 1)
+ self.gly_gbx_exec.addWidget(self.lne_exec_edge, 1, 1)
+ self.gly_gbx_exec.addWidget(self.lne_exec_brave, 2, 1)
+ self.gly_gbx_exec.addWidget(self.pbn_exec_chrome, 0, 2)
+ self.gly_gbx_exec.addWidget(self.pbn_exec_edge, 1, 2)
+ self.gly_gbx_exec.addWidget(self.pbn_exec_brave, 2, 2)
+
+ self.gbx_data = QtWidgets.QGroupBox("用户数据路径", self)
+ self.vly_m.addWidget(self.gbx_data)
+
+ self.gly_gbx_data = QtWidgets.QGridLayout()
+ self.gbx_data.setLayout(self.gly_gbx_data)
+
+ self.lb_data_chrome = QtWidgets.QLabel("Chrome", self)
+ self.lb_data_edge = QtWidgets.QLabel("Edge", self)
+ self.lb_data_brave = QtWidgets.QLabel("Brave", self)
+ self.lne_data_chrome = QtWidgets.QLineEdit(self)
+ self.lne_data_edge = QtWidgets.QLineEdit(self)
+ self.lne_data_brave = QtWidgets.QLineEdit(self)
+ self.pbn_data_chrome = PushButtonWithId("ChromeData", self, "选择")
+ self.pbn_data_edge = PushButtonWithId("EdgeData", self, "选择")
+ self.pbn_data_brave = PushButtonWithId("BraveData", self, "选择")
+
+ self.gly_gbx_data.addWidget(self.lb_data_chrome, 0, 0)
+ self.gly_gbx_data.addWidget(self.lb_data_edge, 1, 0)
+ self.gly_gbx_data.addWidget(self.lb_data_brave, 2, 0)
+ self.gly_gbx_data.addWidget(self.lne_data_chrome, 0, 1)
+ self.gly_gbx_data.addWidget(self.lne_data_edge, 1, 1)
+ self.gly_gbx_data.addWidget(self.lne_data_brave, 2, 1)
+ self.gly_gbx_data.addWidget(self.pbn_data_chrome, 0, 2)
+ self.gly_gbx_data.addWidget(self.pbn_data_edge, 1, 2)
+ self.gly_gbx_data.addWidget(self.pbn_data_brave, 2, 2)
+
+ self.hly_bot = QtWidgets.QHBoxLayout()
+ self.pbn_save = QtWidgets.QPushButton("保存", self)
+ self.pbn_cancel = QtWidgets.QPushButton("取消", self)
+
+ self.hly_bot.addStretch(1)
+ self.hly_bot.addWidget(self.pbn_save)
+ self.hly_bot.addWidget(self.pbn_cancel)
+
+ self.vly_m.addLayout(self.hly_bot)
+ self.vly_m.addStretch(1)
+
+ self.pbn_save.clicked.connect(self.on_pbn_save_clicked)
+ self.pbn_cancel.clicked.connect(self.on_pbn_cancel_clicked)
+
+ self.pbn_exec_chrome.clicked_with_id.connect(self.on_pbn_exec_n_clicked_with_id)
+ self.pbn_exec_edge.clicked_with_id.connect(self.on_pbn_exec_n_clicked_with_id)
+ self.pbn_exec_brave.clicked_with_id.connect(self.on_pbn_exec_n_clicked_with_id)
+
+ self.pbn_data_chrome.clicked_with_id.connect(self.on_pbn_data_n_clicked_with_id)
+ self.pbn_data_edge.clicked_with_id.connect(self.on_pbn_data_n_clicked_with_id)
+ self.pbn_data_brave.clicked_with_id.connect(self.on_pbn_data_n_clicked_with_id)
+
+ self.read_settings()
+
+ def sizeHint(self):
+ return QtCore.QSize(540, 140)
+
+ def read_settings(self):
+ us = QtCore.QSettings()
+ chrome_exec = str(us.value("ChromeExec", ""))
+ edge_exec = str(us.value("EdgeExec", ""))
+ brave_exec = str(us.value("BraveExec", ""))
+
+ chrome_data = str(us.value("ChromeData", ""))
+ edge_data = str(us.value("EdgeData", ""))
+ brave_data = str(us.value("BraveData", ""))
+
+ self.lne_exec_chrome.setText(chrome_exec)
+ self.lne_exec_edge.setText(edge_exec)
+ self.lne_exec_brave.setText(brave_exec)
+
+ self.lne_data_chrome.setText(chrome_data)
+ self.lne_data_edge.setText(edge_data)
+ self.lne_data_brave.setText(brave_data)
+
+ def on_pbn_exec_n_clicked_with_id(self, ids: str):
+ filename, _ = QtWidgets.QFileDialog.getOpenFileName(self, f"选择 {ids}")
+ if len(filename) == 0:
+ return
+ if ids == "ChromeExec":
+ self.lne_exec_chrome.setText(filename)
+ elif ids == "EdgeExec":
+ self.lne_exec_edge.setText(filename)
+ elif ids == "BraveExec":
+ self.lne_exec_brave.setText(filename)
+
+ def on_pbn_data_n_clicked_with_id(self, ids: str):
+ dirname = QtWidgets.QFileDialog.getExistingDirectory(self, f"选择 {ids}")
+ if len(dirname) == 0:
+ return
+ if ids == "ChromeData":
+ self.lne_data_chrome.setText(dirname)
+ elif ids == "EdgeData":
+ self.lne_data_edge.setText(dirname)
+ elif ids == "BraveData":
+ self.lne_data_brave.setText(dirname)
+
+ def on_pbn_save_clicked(self):
+ us = QtCore.QSettings()
+ us.setValue("ChromeExec", self.lne_exec_chrome.text())
+ us.setValue("EdgeExec", self.lne_exec_edge.text())
+ us.setValue("BraveExec", self.lne_exec_brave.text())
+
+ us.setValue("ChromeData", self.lne_data_chrome.text())
+ us.setValue("EdgeData", self.lne_data_edge.text())
+ us.setValue("BraveData", self.lne_data_brave.text())
+
+ self.accept()
+
+ def on_pbn_cancel_clicked(self):
+ self.reject()
diff --git a/da_show_profiles.py b/da_show_profiles.py
new file mode 100644
index 0000000..474bf45
--- /dev/null
+++ b/da_show_profiles.py
@@ -0,0 +1,157 @@
+# coding: utf8
+import os
+from pathlib import Path
+from PySide6 import QtWidgets, QtCore, QtGui
+from util_ext import ProfilesData, DeleteThread, DeleteThreadManager
+from global_vars import accept_warning
+
+
+def sort_profiles_id_func(profile_id: str) -> int:
+ if profile_id == "Default":
+ return 0
+ else:
+ seq = profile_id.split(" ", 1)[-1]
+ try:
+ return int(seq)
+ except ValueError:
+ # if the id is weird
+ return 999
+
+
+class ProfilesModel(QtCore.QAbstractTableModel):
+
+ def __init__(self,
+ profiles_data: ProfilesData,
+ ext_id: str,
+ profiles: list[str],
+ parent=None):
+ super().__init__(parent)
+ self.ext_id = ext_id
+ self.profiles = profiles
+ self.profiles.sort(key=sort_profiles_id_func)
+ self.profiles_data = profiles_data
+
+ def rowCount(self, parent: QtCore.QModelIndex = ...):
+ return len(self.profiles)
+
+ def columnCount(self, parent: QtCore.QModelIndex = ...):
+ return 2
+
+ def data(self, index: QtCore.QModelIndex, role: int = ...):
+ row = index.row()
+ col = index.column()
+
+ if col == 0:
+ if role == QtCore.Qt.ItemDataRole.DisplayRole:
+ return self.profiles[row]
+ elif col == 1:
+ if role == QtCore.Qt.ItemDataRole.DisplayRole:
+ return self.profiles_data[self.profiles[row]].name
+
+ def headerData(self, section: int, orientation: QtCore.Qt.Orientation, role: int = ...):
+ if role == QtCore.Qt.ItemDataRole.DisplayRole:
+ if section == 0:
+ return "ID"
+ if section == 1:
+ return "名称"
+
+
+class DaShowProfiles(QtWidgets.QDialog):
+
+ def __init__(self,
+ browser: str,
+ is_compat: bool,
+ profiles_data: ProfilesData,
+ ext_id: str,
+ ext_name: str,
+ ext_icon: QtGui.QIcon,
+ profiles: list[str],
+ parent: QtWidgets.QWidget = None):
+ super().__init__(parent)
+ self.setWindowTitle(ext_name)
+ self.setWindowIcon(ext_icon)
+ self.browser = browser
+ self.is_compat = is_compat
+
+ self.process = QtCore.QProcess(self)
+
+ # ========== UI ==============
+
+ self.vly_m = QtWidgets.QVBoxLayout()
+ self.setLayout(self.vly_m)
+
+ self.lne_info = QtWidgets.QLineEdit(ext_id, self)
+ self.lne_info.setReadOnly(True)
+ self.vly_m.addWidget(self.lne_info)
+
+ self.trv_profiles = QtWidgets.QTreeView(self)
+ self.trv_profiles.setSelectionMode(QtWidgets.QAbstractItemView.SelectionMode.ExtendedSelection)
+ self.vly_m.addWidget(self.trv_profiles)
+
+ self.pgb_del = QtWidgets.QProgressBar(self)
+ self.pgb_del.setAlignment(QtCore.Qt.AlignmentFlag.AlignCenter)
+ self.vly_m.addWidget(self.pgb_del)
+
+ self.hly_bot = QtWidgets.QHBoxLayout()
+ self.pbn_delete_selected = QtWidgets.QPushButton("删除所选", self)
+ self.pbn_open = QtWidgets.QPushButton("打开", self)
+ self.pbn_cancel = QtWidgets.QPushButton("取消", self)
+ self.hly_bot.addWidget(self.pbn_delete_selected)
+ self.hly_bot.addStretch(1)
+ self.hly_bot.addWidget(self.pbn_open)
+ self.hly_bot.addWidget(self.pbn_cancel)
+ self.vly_m.addLayout(self.hly_bot)
+
+ self.profiles_model = ProfilesModel(profiles_data, ext_id, profiles, self)
+ self.trv_profiles.setModel(self.profiles_model)
+
+ self.pbn_delete_selected.clicked.connect(self.on_pbn_delete_selected_clicked)
+ self.pbn_open.clicked.connect(self.on_pbn_open_clicked)
+ self.pbn_cancel.clicked.connect(self.reject)
+
+ def sizeHint(self):
+ return QtCore.QSize(400, 360)
+
+ def on_pbn_open_clicked(self):
+ us = QtCore.QSettings()
+ exec_path = str(us.value(f"{self.browser}Exec", ""))
+ if len(exec_path) == 0 or not os.path.exists(exec_path):
+ QtWidgets.QMessageBox.critical(self, "错误", f"无法找到 {self.browser} 浏览器的执行路径")
+ return
+
+ indexes = self.trv_profiles.selectedIndexes()
+ cmd = rf'"{exec_path}" --profile-directory="{{0}}"'
+ for idx in indexes:
+ if idx.column() != 0:
+ continue
+ profile_id = self.profiles_model.data(idx, QtCore.Qt.ItemDataRole.DisplayRole)
+ # setProgram 不行,不知道为什么,莫名其妙
+ self.process.startCommand(cmd.format(profile_id))
+ self.process.waitForFinished(10000)
+
+ def on_pbn_delete_selected_clicked(self):
+ us = QtCore.QSettings()
+ user_data_path = str(us.value(f"{self.browser}Data", ""))
+ if self.is_compat:
+ pref_name = "Preferences"
+ else:
+ pref_name = "Secure Preferences"
+
+ indexes = self.trv_profiles.selectedIndexes()
+ total = len(indexes) // 2
+ if accept_warning(self, True, "警告", f"确定要删除这 {total} 项吗?"):
+ return
+
+ del_thd_mgr = DeleteThreadManager(total, self.pgb_del, self)
+
+ for idx in indexes:
+ if idx.column() != 0:
+ continue
+ profile_id = self.profiles_model.data(idx, QtCore.Qt.ItemDataRole.DisplayRole)
+ del_thd = DeleteThread(
+ str(Path(user_data_path, profile_id)),
+ pref_name,
+ [self.lne_info.text()],
+ self
+ )
+ del_thd_mgr.start(del_thd)
diff --git a/daily_check.qrc b/daily_check.qrc
new file mode 100644
index 0000000..6597bc7
--- /dev/null
+++ b/daily_check.qrc
@@ -0,0 +1,9 @@
+
+
+ images/blank_128.png
+ images/browsers/chrome_32.png
+ images/browsers/edge_32.png
+ images/browsers/brave_32.png
+ images/dailycheck_128.png
+
+
diff --git a/daily_check_rc.py b/daily_check_rc.py
new file mode 100644
index 0000000..43a6112
--- /dev/null
+++ b/daily_check_rc.py
@@ -0,0 +1,617 @@
+# Resource object code (Python 3)
+# Created by: object code
+# Created by: The Resource Compiler for Qt version 6.6.1
+# WARNING! All changes made in this file will be lost!
+
+from PySide6 import QtCore
+
+qt_resource_data = b"\
+\x00\x00\x01\xb8\
+\x89\
+PNG\x0d\x0a\x1a\x0a\x00\x00\x00\x0dIHDR\x00\
+\x00\x00\x80\x00\x00\x00\x80\x08\x06\x00\x00\x00\xc3>a\xcb\
+\x00\x00\x00\x01sRGB\x00\xae\xce\x1c\xe9\x00\x00\x00\
+\x04gAMA\x00\x00\xb1\x8f\x0b\xfca\x05\x00\x00\x00\
+\x09pHYs\x00\x00\x0e\xc1\x00\x00\x0e\xc1\x01\xb8\x91\
+k\xed\x00\x00\x01MIDATx^\xed\xd21\x01\
+\x00 \x0c\xc0\xb0\x81\x7f\xcf\xe3\xc1E\x9a\xa7\x06zv\
+w\xe2\xba\xbfA5\x00\xae\x01p\x0d\x80k\x00\x5c\x03\
+\xe0\x1a\x00\xd7\x00\xb8\x06\xc05\x00\xae\x01p\x0d\x80k\
+\x00\x5c\x03\xe0\x1a\x00\xd7\x00\xb8\x06\xc05\x00\xae\x01p\
+\x0d\x80k\x00\x5c\x03\xe0\x1a\x00\xd7\x00\xb8\x06\xc05\x00\
+\xae\x01p\x0d\x80k\x00\x5c\x03\xe0\x1a\x00\xd7\x00\xb8\x06\
+\xc05\x00\xae\x01p\x0d\x80k\x00\x5c\x03\xe0\x1a\x00\xd7\
+\x00\xb8\x06\xc05\x00\xae\x01p\x0d\x80k\x00\x5c\x03\xe0\
+\x1a\x00\xd7\x00\xb8\x06\xc05\x00\xae\x01p\x0d\x80k\x00\
+\x5c\x03\xe0\x1a\x00\xd7\x00\xb8\x06\xc05\x00\xae\x01p\x0d\
+\x80k\x00\x5c\x03\xe0\x1a\x00\xd7\x00\xb8\x06\xc05\x00\xae\
+\x01p\x0d\x80k\x00\x5c\x03\xe0\x1a\x00\xd7\x00\xb8\x06\xc0\
+5\x00\xae\x01p\x0d\x80k\x00\x5c\x03\xe0\x1a\x00\xd7\x00\
+\xb8\x06\xc05\x00\xae\x01p\x0d\x80k\x00\x5c\x03\xe0\x1a\
+\x00\xd7\x00\xb8\x06\xc05\x00\xae\x01p\x0d\x80k\x00\x5c\
+\x03\xe0\x1a\x00\xd7\x00\xb8\x06\xc05\x00\xae\x01p\x0d\x80\
+k\x00\x5c\x03\xe0\x1a\x00\xd7\x00\xb8\x06\xc05\x00\xae\x01\
+p\x0d\x80k\x00\x5c\x03\xe0\x1a\x00\xd7\x00\xb8\x06\xc05\
+\x00\xae\x01p\x0d\x80k\x00\x5c\x03\xe0\x1a\x00\xd7\x00\xb8\
+\x06\xc05\x00\xae\x01p\x0d\x80k\x00\x5c\x03\xe0\x1a\x80\
+6\xf3\x00\xedw\x03\xfd\xcd\xd11\xb4\x00\x00\x00\x00I\
+END\xaeB`\x82\
+\x00\x00\x0el\
+\x89\
+PNG\x0d\x0a\x1a\x0a\x00\x00\x00\x0dIHDR\x00\
+\x00\x00\x80\x00\x00\x00\x80\x08\x03\x00\x00\x00\xf4\xe0\x91\xf9\
+\x00\x00\x03\x00PLTEGpL\xd3\xb0\x22\xfb\xd1\
+*\xe5\xbf%\xd3\xc1'\xf9\xd1(\xdf\xba$\xe8\xca*\
+\xd7\xc4)\xd8\xc2'\xd8\xb4$\xd2\xaf#\xfa\xd1'\xfd\
+\xd4-\xf4\xcd&\xd1\xaf#\xcf\xad#\xd0\xae\x22\xf9\xcf\
+'\xfa\xd0(\xf6\xcf&\xca\xb0#\xcd\xab\x1f>\xb56\
+\xfa\xd1'\xfb\xd1'\xfb\xd2)\xfa\xd0&\xd3\xb0#\xd0\
+\xae&\xf8\xce%\xf9\xcf'\xf8\xcf&\xf5\xcb#O\xcc\
+?\xd0\xae\x22\xd5\xb2%\xd3\xb0&\xcf\xac!\xfb\xd1(\
+\xfb\xd1(\xfd\xd3+\xfb\xd2)\xfd\xd4-\xf7\xcd\x22\xf9\
+\xcf%A\xbe9\xd3\xb0$I\xca@O\xbf7\xfd\xd3\
+,\xf6\xcc \xf7\xce(\xfa\xd1'\xd1\xaf#\xd3\xb0%\
+\xcf\xad \xd3\xb0#\xd1\xaf'\xd3\xb0&@\xbc9\xfb\
+\xd1(\xf9\xcf$\xfd\xd3,\xf8\xce\x22\xda\xb7'\xd4\xb2\
+'\xd8\xb5&\xd3\xb1'\xd0\xae\x22\xcc\xaa\x1e\xca\xb3\x22\
+\xfe\xd4-\xf3\xca#\xfd\xd4,\xfd\xd3,U\xdeK\xe2\
+\xbd(\xfd\xd4-P\xd7F\xe0\xba%S\xdcHZ\xdc\
+J=\xb46D\xc4;=\xb75<\xb25\xf7\xcd!\
+L\xd6B\xef\xc6 T\xddJO\xdbF\xf6\xcc \xd9\
+\xb5$C\xc0;A\xbc:\xf3\xc9!\xf7\xcd\x22\xd8\xb5\
+&<\xb35W\xdfM\xd6\xb4*X\xdfN\xdd\xba*\
+\xf1\xc8$C\xc1;\xf7\xcd\x22V\xddJ\xb7\xbd(\xfd\
+\xd4-G\xc7>>\xb56V\xddKd\xdcF\xed\xc4\
+\x1f\xff\xcf'\xff\xcf'\xfb\xd1)\xfb\xd2*\xff\xff\xff\
+\xf9\xcf&\xfa\xd0'\xfa\xd1(\xfc\xd3,\xf7\xcd#\xfc\
+\xd3+\xfc\xd2+\xf7\xcd\x22\xf8\xce$\xf6\xcc!\xf5\xca\
+\x1e\xfa\xd0(\xf6\xcb\x1f\xf9\xcf%\xf9\xd0'T\xddJ\
+\xf5\xcb\x1f\xf4\xca\x1d\xfd\xd4-\xf4\xc9\x1c\xf6\xcc \xf8\
+\xcf%P\xdcGW\xdfNV\xdeLK\xd9AI\xd8\
+?\xf3\xc8\x1a\xf8\xce#S\xddIR\xdcHE\xd6<\
+U\xdeKP\xdbFF\xd7=G\xd7>D\xd6;O\
+\xdbEH\xd8?\xf4\xc9\x1dN\xdbEC\xd5:B\xd5\
+9\xfd\xd3,\xfd\xd4.\xf7\xcc\x22\xf7\xcc!L\xdaC\
+\xd1\xaf'\xfd\xd3-G\xd7=C\xd6:\xcb\xa9\x1eM\
+\xdaD\xfe\xd5.M\xdaCY\xe0OY\xdfO\xcf\xac\
+!@\xd47A\xd58\xdf\xba&\xf5\xcc&F\xca>\
+A\xca8\xfe\xd4.\xeb\xc4'\xda\xb6$\xd9\xb4\x1e=\
+\xb65\xd4\xb0\x1e\xfc\xfe\xfc\xe6\xc0(\xe9\xc1\x1d\xef\xc5\
+\x1cK\xd6BC\xd0:\xdf\xb8\x1f\xef\xc8$\xd4\xb1&\
+\xfa\xfc\xfa\xf0\xc9)N\xc9F\x8d\xd36H\xd1?\xe2\
+\xbb\x1eB\xc58\xe7\xce%\xed\xed\xed\x9c\xeb\x97z\xe4\
+s\xe2\xe6\xe2\xf7\xf7\xf7>\xc06\xc2\xf2\xbe\x9b\xd69\
+\xd3\xd0*Y\xc9R\xbe\xd8\xbd\xf4\xf4\xf4\xf1\xfc\xf1\xc8\
+\xd2.\xdd\xf8\xdby\xdbD`\xddI\xf6\xce)z\xcb\
+u\xae\xd4\xab\x88\xcd\x83\xfe\xd5/\x87\xe7\x81\xb4\xd31\
+\xdf\xcd%\x9c\xc3+\xcb\xdb\xcae\xc9^\xa1\xd1\x9e\xaf\
+\xef\xab\xbf\xd2/j\xdbDm\xe2ec\xdeZ\xcd\xc4\
+!\xd8\xe0\xd7q\xcbk\xb6\xef\xb2\x84\xe2}X\xd7P\
+\x98\xd0\x94[\xc16\x8b\xb6+l\xc97\x8e\xe7\x88o\
+\xb0.\xfb2\x02\xfb2\x02\x90\x0aZ\x1e\x00\x00\x01\x00\
+tRNS\x00!s\x17\x05\xcf\x0b\x07\x02\x01\x13n\
+!\xf3\x1b\xd6\xcc\x8bH6S-^\xb8\x10\xab\xb9\xa1\
+\x98\xf4h/@{\x0c\x81K\xb3R\x5c\xf0\xc0\x0e\xfa\
+\xf0\xb1Dd\x1f\x15\x83\xfb(\xfa\xc0vD4\xfa\xed\
+\x8a\xe0\xc7\xd8\xf6\x8d\xe3X\xa3\xaa\xe9@\xeab\xdd\xe6\
+\x93\xfc\x92(\xc6L\xcd\xc9vR\xd2\xe8\xaf\xa1\xea\xab\
+\xe5\xf3\xa5\x96\x90\xdd:\xec\xe5\xc4\xf5\xf9\xc2\xe8\x90d\
+\xfe\xdc\xda\xf1\xd7\xb8\xf7\xff\xff\xff\xff\xff\xff\xff\xff\xff\
+\xff\xff\xff\xff\xff\xff\xff\xff\xff\xff\xff\xff\xff\xff\xff\xff\
+\xff\xff\xff\xff\xff\xff\xff\xff\xff\xff\xff\xff\xff\xff\xff\xff\
+\xff\xff\xff\xff\xff\xff\xff\xff\xff\xff\xff\xff\xff\xff\xff\xff\
+\xff\xff\xff\xff\xff\xff\xff\xff\xff\xff\xff\xff\xff\xff\xff\xff\
+\xff\xff\xff\xff\xff\xff\xff\xff\xff\xff\xff\xff\xff\xff\xff\xff\
+\xff\xff\xff\xff\xff\xff\xff\xff\xff\xff\xff\xff\xff\xff\xff\xff\
+\xff\xff\xff\xff\xff\xff\xff\xff\xff\xff\xff\xff\xff\xff\xff\xff\
+\xff\xff\xff\xff\xff\xff\xff\xff\xff\xff\xff\xff\xff\xff\xff\xff\
+\xff\xffss\xaa\x1e4\xff\x00\x00\x0a\x1bIDAT\
+x\xda\xbd\x9byX\x14G\x1a\xc6\x9bc`\x86C\x06\
+\x04\xb9QNAP\xc1\x03T\xf0\xbeWQ\xe3\xb9f\
+\xe3\x15s'\xee\xf5A\x83\x18Y\xf1\
+X\xa9[\xc0=(\xc8\xcbK\xe8\xbd0`\xbc\xadJ\
+|\xfc\xac\xb1\xe3e\xb2\x84\xf87\xa5A6N&\x1b\
+;6>~d\xe7u\x88\x172\x09\x1f\xf9wq\xb4\
+7\x9a25\x06\x8d&\xd3\xd4\xb8Yd\xd5\xb1\xa8\xf5\
+R\xb08u\x93\x81d\xa0\xd1\x8b\xf3\x80\xd8Oq\x8b\
+\x17Q\xbfBC\x89\x86\xa7\xfazN\x8e\x89\xe9\x97\x16\
+\xa6\x1cC\x92\x0d\xa6v\x11\x01\xc6\xf1\x0d\xf8\xff\x0b\xcc\
+:\x13I\xa6\x87\x8dr'\x14\x1dX\xdc\xc7M\x1dC\
+Y\xd8\x0c\x8f\xc1\xdfO\x09\x05\xd7ml 'y\xb2\
+\xdf\xaf\x94\x0f\x17\xdfp\xd2P\x01\x0e\xc2\xdbOq\x8f\
+\xff\x1b\x10=9&\x1a\xf9zw\x99B64B\xa3\
+\x84p\xa7\x00?\xb8\xbe\xcfh\xa1\x89$.\x9d\xac\x07\
+\x86\x19\xc9}\x0eB\xc1\xfa\xb1\x83\x84\xa72'\x1f\x12\
+\xda\x0639\x1b1\xc0\x1e\xa8'\xa7\xd8{\xb7+<\
+\x94\x0d\x15\xb0H\x9c=\xad\x01\x9bAT4\xc4b\x96\
+\xb9\x1e>\x86u\xa0P#Y\xe3H\xb1\x12f@\xa3\
+\xc4\xae1\x9d\xd2M\xb0X\xfe\xec\xd4\xa3b\x1d\x80\xfa\
+p'\xfc+-\x86\x04\xc5\xda\x1e\xc6\xaa\xe5k\x80\xd4\
+2\x80\x16XS4\x10\x03\x1a_v\xc3\x85\x93\xa6\xed\
+\xd8\x06P\x82\xf69<\xc2\xf17Sa 9\x0f\xf3\
+\xa8I$\xb6\x114\xc0\xec:\xda\x84\xd3\xd7\x93\xa4/\
+\xcf\xf7T\x92\xd4\xdb\xad\xb5\xc1\x07\x98\xdby,\xc3\xdc\
+\xbe\x86LG\xad\xd5&\xa7\x93\x9a\x8a\xb7\x84\x81/\xb1\
+\xa7\xd8\x0b\xf3\x96\xbe\x81\x8cE'E.\xb1d\x83^\
+\xb8\x22|\x93#\xc6N\x14\xea\xf6\xc3\x87\x0a5\xa5b\
+h\xb8\x9dF\x98\x07\xdf\xe0\xce\x14\xd4\xafo S\xed\
+=\xcbN\xa9dC\xfdv4\x22\xd67\xf3\x04Bd\
+S\xb7\x9ff\xffQ\x92\xa7\x8d\xa1\x1aa\x03\x0a\x11k\
+\xfc\x89\xc8\x00\x1b\xa8\xdbW\x06\xe0wF\x94\xa4\x06U\
+;\xbb?\xdc\xc0L\xa4~\x05\x95I@\x96\xa9\xae\xd1\
+\x06\xa4\x01\xf86\x8f\xdc\x0f\xdd\x00\xc2\x99\x04\x87Hd\
+}\x11\xbb\x1c\x03\x90\x01\x92<\xa0\xf5\xfd\x91\xf5\xdd\xbb\
+k`\x15t\x91\xda\x7f\x15\xb2\xbe\x88\xc3\x17t\x17l\
+x\x0dt\x0f\xf2\xc8\x09\xe8\xea\x22v{\x07\xa2#l\
+\x90\x01\x06\xb2\xd7\xb8=\x02\xf8\xc1\x0dD\x0a\xc5\x98\x10\
+\x89y\xa1*\x22&\x08\xd5\xdd#\xe2x:X0\xc8\
+\x9eq^vw\xe6B\xb2\x85\x11q4\xfa\x9c\x9d0\
+\x13\x22\x84\xc7\xe2\xc0$;\x15\xb3\x93\xa0{]\x8a\x11\
+\xf6\xc2d\x7f\x10\x22p\xce\xed\x12\x9cm\x1f\xf0 \x88\
+\xb4\x1f''\x09y\xa0\xea\xbf\x04\xa3\x9f\x1d\x0c\x9d\x08\
+W\xe1\x22\xe5\x04\xbb\xf0\xcf(\xd7c\x19\xee\x05\x9c\xc7\
+\xa8\xef\xe6\x9c\xb9\xf4\x15\xc5\xa5\xbb9\xe8XK8'\
+y\xee\xb2\xf5\x00Ba\x0d {\xd0v\xa3\xec\xb3N\
+\xcan\xb4\xdd9\x83\x88\x95\xc4n\x83\x08\x88\xfe\xfal\
+\xc0~W\xd4\xeb\xf3iq\x9a\x1b\x08\x0f\xac\xa7Z\xbe\
+\x14d`\x9b\x0c7\x97&\xcf~\xa9\x0cM\xdb]v\
+\xac\x1c\xf6\xc2dj\xce\x07 ^\xb3\x7f\xf7\xb3_\xda\
+)HY\xdb\x19f$}\x1a{c\xbc\xe6C\x183\
+\xed\xe8?\xf3\xc2N{\x94\xdd|\xc0\x08T\xc7\xce\x0c\
+=\x0c90\x86\xfb\x0b\x1e6\xcf\xde\x89\xa5\xa9\x9e\x0e\
+\xe4\xc3\xa9\x9e\x9a\x07u \xd0\x06\xc9\xf3\xf1\xfa;?\
+{x\xd7\x1a%\xaf>\x9a{8Q\x93\x03e%j\
+J\x9e\xfb\xc2.\x08e7/Y\x83h\xb8\xc9\xa9\x8b\
+\x01l g\x1c\xef\xad\xae\x98\xfb\xec. V\x07\x1f\
+\xfa\xf0n!5o\x1b\x98\xe1K9\xaf\x93d\xb0~\
+\x87\x03*B=omJx\xd6\xc0\x0dP#!t\
+@Wf\xa1\x88\xfa\xf3;\x22xxf\xdb\xb6\xd3\xfc\
+\xfc\xd8\xc9\xb0M\x1cKB#\xfa\xbaP\xafv\xd7>\
+\xd3\xff\xf2\x8e(n\xe4\xe5\xa1\x0eQ\x94\xa7\xf36\x89\
+'i\xc2\xa6Mwr\xc5\xb1\xab\xcd\xe0\x8b:\xa4+\
+\xaf\xaa\xb3`0TVVuP^\xf9\x1e\xc4\xc3\xdd\
+\x9b\xb9by\x03\xb5@S\x84\x95\xf3\xa8\x01\xe8\xe74\
+\x89\xd6\xcf\x9d\x8f\x9eKF\xa7EG{z\x0e\xed\x17\
+\x17\x173}z@\x80\x93\xd3\xe4\xbf\x02\x0c\x5cz\x1b\
+\xc6\xe3s\xe7m\x7f\xee\xcd}\x06\x92\xd9u\x1cZ\xe5\
+a\xd9x\x03$\x9f\xfb\xe8\x9a\xaa\xfa[\xda\xc1|`\
+r\xb5\x0ao\xe0\xd2^\x10\xe7\xaf\xa9T\x94\x03\xfaz\
+\x08\xec\xb7\x5cx\xfdMMp}\x95\xea\xe4O\x9d\xd7\
+o\xff\x01d`%\xde@}.H\xff\xa4\xca\xca\xc9\
+\xf3\xb6\xa2\x14Hr\xf7\xdcF,\x0fv\x00\xf8\xd6\xa6\
+O9xd)\xd9\xbb\xe3u\xc8\x1a\x17\xaf\xbf\xb1I\
+\x9c\xbe\xaa\xfa\xa7\xceBH\x1f,\x05\x18x\xb8\x0f\xcb\
+\x7f\xab\x19\xfa\xf7~\xb6\x16\x96<\x8b\xdf,pK\xc2\
+\xeb\x9f\xc1\xeb\x9fc\xe8\x9f\xbcw\xb9\xd4V>\x17\xbf\
+\xbe\xc0\xebo\xfd\xaa\x84\xc1\xbeG\xbbJx\x9cS1\
+\xf4\xaf\x5c>A\x7f\x80\x9f\x8b\x06o\xc1s\x87!U\
+v]u\xed\x8c\x5c>\xb5\xd0\xc6\
+w\xcc\xa0\xcd\xa7\xbe?\x0b\xd4\xdf\xbd\x02v\x04\x19\xab\
+.\xaf\xa1dj\xaa\xd4S\xddX\x0bI\x9b\x81'\xcd\
+,\x07\xda[\x17X\xddp\xf6\x96\xf6\x00R\xbf\x19\x96\
+\x95\x12\x0aOj\xde\xa9\xacR\xab\xd9\x87\x96Nut\
+\x13\xfcz\x80I\xb1\xf6\xfe=F7\x5c\xb8\xa5->\
+\x80\xe4U\xf0o\x1c\x03|\xd4j\xf54\xce/\x02\x15\
+>]}\xc0Q\xa8et\xc3\x85\x1f\x84\xf4\x0fd\x10\
+`\x5cc<\xe3x\xc7E\xbef\x9b\x81\x1am1\x9b\
+f\xed\xed\xcen\xe8\xd0\x17@\x9b,\xe6g\xe6\xa8\xe1\
+y\x9an\x82\xdf\x8a8\xc1\xdf\xb7vC\xf5\x85\xfb\x82\
+\xfaE\xcb\x89n\xe2\xba\x8c6p\xb7\xe5c\x0e\xfbk\
+O}}\xf6\xea\xd3\xfb\xda\x8f\x85h\x99\xdb]\x03\xc4\
+Ts\xd7(\xa8\xdd\xcf\xa1\xb8V{\xeb\xf6e^\xb1\
+\x8d\xa2\xe6\x15\xdd\xd6'\xfau=\x07\xe6_\x8b\xf8\x1a\
+:\x9d\xa0\xfe\xfe\xa2W\xa3\xbao\xc0\x83\x9e\xa2\x0a\xcd\
+OZ\x8ax\x14\x17\x09\xd32\x84\x90\x00j\x9e.\xaf\
+\xaa\xaa\xb2l\xa6\xfc\xd6R$\x02]\x86\x14\xfa\x0a\xd7\
+hu\x17\xff\xd3\xbd\x0bF\x17(\xd5?\xc8\xba\x8d\x1e\
+\x15\xd7\x8f\x22nz@J \xd8\x81nA\x14!=\
+\xf2\xa8@\xdd' t\x0bR\x88\x1e!*\xb0\xf5\xd0\
+\xfbX\x0e\xb5\x04F\x11=\x84<\xc3X\x8b\xd3\xafm\
+]\xd1\x93\xff\x9e\xbchu\xcb\xe7\x87\xec\xd1\xf2\xa7E\
+D\x8f\x92\xb2\xdc\xa8\x13\x96\xd7\x19\x03\x93\x89\x9ef\xc8\
+\x02\xa3\xees$:\xe3\xe2!Do0$\xd0\xd8\xaa\
+;\xce\x16?\xaek5\x06.R\x10\xbdD\xf2\xc2\x05\
+Fck\x8b\xae\xf68E\xad\xae\xa5\xd5h\x5c\xbc0\
+\x99\xe8UR\x16e\x04.^m\xa4X\xbd80c\
+Q\x0aa\xe7\xee\xff\x0fO2\xbfP\x8b\xfe4z\x00\
+\x00\x00\x00IEND\xaeB`\x82\
+\x00\x00\x05~\
+\x89\
+PNG\x0d\x0a\x1a\x0a\x00\x00\x00\x0dIHDR\x00\
+\x00\x00 \x00\x00\x00 \x08\x03\x00\x00\x00D\xa4\x8a\xc6\
+\x00\x00\x00\x09pHYs\x00\x00\x0e\xc4\x00\x00\x0e\xc4\
+\x01\x95+\x0e\x1b\x00\x00\x02\xbePLTEGpL\
+\xd7`*\x17\x0a\x04\xa0G\x1fO#\x0f\x04\x01\x00\xf7\
+q4\xf6n0\xdfd,\x00\x00\x00M\x22\x0f\x14\x09\
+\x04\x02\x01\x00\x9eG\x1f\xd8a*\xddc+-\x14\x08\
+Q$\x10\xb5Q#\x22\x0f\x06\x06\x02\x01\x17\x0a\x04\xa1\
+H\x1f\xf7}E\xdbb+\xf6n0\xf5n0N#\
+\x0f\xafN\x22\x15\x09\x04_*\x12\x02\x01\x00\xa9L!\
+\xe4f-X'\x11}8\x182\x16\x09&\x11\x07\xe0\
+d,\x9fG\x1f\x09\x03\x01\xe3f-\x0c\x05\x02b,\
+\x13\xb8R$m1\x15Q$\x10b+\x13z7\x18\
+\x04\x01\x00W'\x11\xf8}E\xf4\x85S\xf7\x86R\xa1\
+H\x1f\xa7K!\xbeiB \x0e\x06)\x12\x08\xd5u\
+I\xd3_)\xd5`*W'\x11\xcdrH\xc6mD\
+\xf2\x82O\xf7~G\xa0G\x1f\xf7x>\xb6e@\xcc\
+\x5c(\x98D\x1e\xf8\x7fHK!\x0e\xe6|K\x97R\
+4\xf7w=\xcf])\xf6\x83O\xf7|C\xf7v;\
+\xf8\x80I\xee\x82Q\xf7|D\xe8\x7fO\xf7\x7fG\xd0\
+tI\xe4~O\xe8h.\xddxI\xf7u:\xdf{\
+N\xe9i.\xf7y@\xf7y@\xf7o1\xff\xff\xff\
+\xf8}E\xf6n0\xf5n0\xfe\xfe\xfe\xfe\xf0\xea\xea\
+i.\xf7p2\xfa\xf0\xec\xfc\xf6\xf3\xf7x>\xfe\xf3\
+\xee\xe3{K\xdfg0\xf8\x8d\x5c\xfb\xbd\xa1\xf6\x7fI\
+\xfc\xce\xb9\xf8\xeb\xe5\xfe\xfc\xfb\xf4m0\xe6\x83V\xe3\
+\x81T\xe3\x81U\xfb\xf5\xf2\xf9\x9cr\xe2\x89`\xfb\xf5\
+\xf3\xfb\xc9\xb2\xf1\xa8\x86\xefm1\xf0\xd4\xc7\xfc\xf5\xf1\
+\xe5r>\xfd\xf9\xf8\xf9\xf0\xeb\xe6\x82U\xe5\x90i\xfc\
+\xd8\xc8\xf6o1\xe5r=\xf6n1\xf1\xd6\xc9\xf9\x9a\
+n\xfd\xfa\xf8\xf1\xa5\x82\xe4\x91j\xf0k/\xe2\x89a\
+\xf0l1\xfb\xc7\xaf\xe9\xba\xa4\xf0l/\xe3\x97t\xe2\
+|M\xfc\xd6\xc5\xe4g/\xf7\xe8\xe1\xf8\xeb\xe6\xf9\xc2\
+\xa8\xe2}N\xf8\x81K\xe4\x9ay\xf7v<\xf8\x85Q\
+\xe9\xb8\xa2\xe4g.\xfe\xfa\xf8\xf7v;\xfe\xf7\xf4\xfc\
+\xf7\xf4\xf8\xbd\xa3\xf7\xe7\xe1\xfe\xfa\xf9\xe4\xaa\x8f\xf8\xec\
+\xe6\xfa\xae\x8c\xf7o2\xe5\xac\x92\xfe\xf9\xf7\xf7{C\
+\xfc\xcd\xb8\xfa\xa6\x80\xfa\xb7\x99\xfc\xd2\xbf\xf7zB\xf9\
+\xa3|\xfd\xea\xe1\xf5\xe3\xdb\xfd\xec\xe3\xfc\xd7\xc6\xfd\xdd\
+\xce\xfa\xaf\x8d\xfe\xf2\xed\xfc\xd9\xc9\xfa\xae\x8b\xe1\x96s\
+\xe3\xa9\x8e\xfe\xf2\xec\xdfxH\xfd\xfa\xf9\xe1\x97t\xdf\
+vF\xf6\xe7\xdf\xf9\xa4}\xe3\xa8\x8d\xfa\xf1\xed\xfe\xee\
+\xe7\xe2\x8bc\xfa\xb5\x96\xfe\xfc\xfc\xe2\x8ce\xfe\xed\xe6\
+\xfd\xfb\xfa\xf2\xad\x8d\xfd\xdf\xd1\xfe\xf8\xf6\xfe\xef\xe9\xe2\
+\xa5\x8a\xed\x95l\xf1\x82O\xf4\xc2\xab\xf9\x95g\xf8\x8c\
+Z\xed\x97p\xef\x98q\xe2\xa3\x87\xf8\x8c[\xe0\x89b\
+\xfd\xe0\xd2\xf1\x81N\xf2\xac\x8c\xf9\x96i\xf4\xc3\xad\xfc\
+\xcf\xba\xef\x99q\xfe\xf9\xf6\xe0\x89a\x19\x07\xe4\xa6\x00\
+\x00\x00_tRNS\x00\xe1$\xb1c\x08\xfc\xfe\xe9\
+\x12a#\x07\xaf\xe2\xe2\x03d\xa0\x02\x05%\xb2\xfe\xe1\
+\xfd\xfdb\x94$D\x08\x93\xebC\x89\x06\x05\xea\xb0\x02\
+\xea\x028\xa2CeB\x87\x056\xfeY[\x88\x89\x11\
+\x01\x01\x14\xd7\xd9-;\x13\x5c\xcc~\xe3\x10\xce|\xcf\
++2\x01\xe4\xcf^\xc5\xf6\xcd9\xc37\xca<.\xe0\
+1\xf7-\xdf\xe8\xe7\xfe\x08\x05\xdf\x00\x00\x01\xfbID\
+AT8\xcbc`\xa0\x220\x11O\x84\x02cSl\
+\xf2!\x01\xf1p\x10\x15\x84.\xeb\x1a\x16\x1d\x17\x8f\x04\
+b\xc3\xed-\x90\xe5\x1d\x02\x97d\xc5\xa3\x80\xac\xf9\x1e\
+\xd6H\x0aB\xe3\xb1\x00'$\x051\xf1\xf1[\xb7\xef\
+\xdb\x9f\xb7mcvv\xe1\x89\xbc=\xbbwn\x88\x8f\
+\x8f@\xc8G\xb2\xc5\xc7oI@\x01\xbb\xd6\xc6\xb3y\
+\xc1\x15\x04\xc7\xc7o\x02\x0a\x16\xa7@e\x8f\x00\x19k\
+\xe2\xe3\xfd\xbd\xa1\xf2\x8e\xe9\xf1s\x97&\x1c?\x9a{\
+\xecdQBB\xd1\xc1\xc3\xb9\x07v$,^\x10\x9f\
+\xee\x07U\xe0\x12\x1f?\x1d\xa4\xf1P~\xc1\xe3\x84\x84\
+K\x05\xf9{A\xbc\xb6\xf8x\x1f\xa8\x02\xb7\xf8\xf8\x06\
+\x88\xd9\x0f\x0b\x13\x12\x9e]\x86\xb03\xdb\xe3}\xa1\x0a\
+\xdc\xe3\xe3[\x9b\xc0\x82O\xaf&$\xa4\xde\x043{\
+\xfa\xe2\xe3\x9d\xa1\x0a<\xd9\xe3\x93\xab*@\xa2wA\
+\xc4s\x10Q=!9\x9e\xdd\x0e\xaa\xc0\xd2&>>\
+\xa9\xac9!!\xe5\xc6\x83{\xd7^>:\x97\x90\x90\
+Q\x9e\x14\x1fok\x05\xf3\xa6\x190\xdc\x92Z\x80\xfa\
+R\xef\xc7\xc7\xdfN\x052J\x81\xf2\xf1\xe6\xf0p0\
+\x02\x85l7\xc8\xe4\x17On]\x01\xd1\x95 \x11m\
+\xb8\x02a\x09 w\x12H\xa2\xf8\xce\xf5\x8b \xba\x06\
+(\xc0/\x80\x08k\x05 \xbf\x17(~\xea|I\xc9\
+\x99\x0c \xa3\x16( \x83\x14Y\xfa@\xfe\xc4\x85\xab\
+\xd7\xe5\x9c\xbdpzs\xce\xaaY\x93\xeb\x80\x02\xbaH\
+\x0aT5\x81\x02S\x16-_\x0f2~\xe5\xec\x19\xfd\
+@\xae\xa2\x0ar\x8a\x11\x029*i\xde2p\x10N\
+\x05y!^\x07%\xc9)I\x81\xc4\x92\xe7\xacHH\
+\x989-\x19\xc4\x96\x94CQ\xa0a\xc0\x0e6\xa3\xa3\
+\xbe\xab\xb1\x13\xc4bW\xd6BM\xb5\x86,\x1c\xe0t\
+\x96\x96\x06\xa6\xd4\xf4D0\x92\xbd\xac:;,5\xb2\
+\x8b\x89b\xcb\x18\xf2\xd2\x10C\xe29\x04Yq\xe4-\
+n^\xa0!\xec\xccL\xb8s\x1f\x0f\x17#\x1f\x0b+\
+\xde\x0c\xca\xc9Ij\x96\x06\x00\xfd\x98\xe6y\x07\xbeQ\
+(\x00\x00\x00\x00IEND\xaeB`\x82\
+\x00\x00\x05}\
+\x89\
+PNG\x0d\x0a\x1a\x0a\x00\x00\x00\x0dIHDR\x00\
+\x00\x00 \x00\x00\x00 \x08\x03\x00\x00\x00D\xa4\x8a\xc6\
+\x00\x00\x02\xe2PLTEGpL\xfc\xceI\xfc\xce\
+I\xec\xc1D\xdcJ=\xdcJ=\x0f\x93S\xfc\xceI\
+\xfc\xceI\xfc\xceI\x9e5,\xdbI<\x0f\x98V\xf4\
+\xc7F\x0cvC\x96{+\xdcJ=\x0buB\xf6\xc9\
+G\x10\x9dY\xdbI<\xcf\xa9<\xdbI<\x10\x9dY\
+\xdcJ=\x9f\x82.\x0ac8\x86-%\x9e5,\x10\
+\x9dY\xbc\x996\xdcJ=\x0f\x99V\xbd\x9b7\x10\x9d\
+Y\x0d\x81I\x0f\x98V\x10\x9dY\xb7=2\xdcJ=\
+\x10\x9dY\xb7=2\xf5\xc8G\x09]5\xdbJ<\xdc\
+J=\xb6=2\xdbI<\xaa\x840\xf0\xa0D\xdaJ\
+<\xb6=2\x0bk=\x10\x9dY\xdcJ=\xc3A6\
+\x5c9(\xdcJ=\x10\x9dY\xc3A6\xdcJ=\x18\
+\x98W\x10\x9dY\xdcJ=\xdcJ=\xf1\xc5E\x0f\x9b\
+W\xf9\xcbH\x10\x9dY\x9d4+\x0ac8\xdcJ=\
+\xd9I<\xc3A6\xd9I<\xc3A6\x9d4+\x10\
+\x9dY\xfc\xceI\x0d\x82J\x0e\x91R\x0d\x80H\x0e\x90\
+R\x0e\x95T\x0ah;\x8f\xcc\xae\xeb\xed\xee\xfc\xceI\
+4\xa1hE~L\x13\x9bX\xa5\xa2\x8c}gC\xa0\
+\x83.`rG\xa8E3\xbdA3\x1d\x96V\xe8\xbd\
+C\xd1\xab<\xa7\x880\xcd\xa8;\xea\xbfC\x0f\x9cX\
+\x0f\x9cX\x94y+\x09\x5c4\xbf\x9c7\xdc\xb4?\x0c\
+wC\x10\x87L\x9a\xa6D\xd9\xb1?\xdcJ=F\x87\
+\xf3\xfc\xceI\x10\x9dY\xfb\xcdH\xc7\xd7\xf1\xe9\xb5\xb0\
+\xddTH\xde\xe5\xf1F\x87\xf2\x0e\x94T\x0f\x9bXG\
+\x85P\xed\x96?\xeb\x94<\xb9WA\xee\x9aB\xdbI\
+<\xf6\xde\x97\xf9\xc9D\xdaH;\xf6\xde\x99M\x8b\xf2\
+\x15\x9e\x5c\x9b]A\xb4\xda\xc8\xdfcX_\x96\xf2\xc1\
+\xd3\xf1\xf6\xc2>\xc0\xd2\xf1\xcf\xdc\xf1\xf8\xd9\x81\xef\xef\
+\xf0\xea\x91:\xe5\x96\x8f\xfb\xd0S\xf8\xd8|\x84\xc8\xa7\
+\x16\x9f]\xd9G:\xcc\xda\xf1/\x8eS\xe9\x908\xe8\
+\xae\xa9\xaa\xc5\xf1\xe8\x8e7\xf2\xba7\xe8\xb0\xa7`\x97\
+\xf2\xcdA5\xe1\xe7\xf1\xf1\xef\xea\xfb\xcfO\xe6\x91I\
+\xe7\x8b4\xad\xc6\xf1\xc2\xd4\xf1\xf0\xef\xed\xef\xd5\x9b\xee\
+\xb45\xe8\xa5\x86\xd8\xe2\xf1\xd0\xdd\xf1J\x89\xf2\xd8F\
+:\xdb\xe3\xf1\xf0\xb73\xf4\xbf>\xe7\x8c5U\x90\xf2\
+\xa1\xbf\xf1\xe6\x8a3\xed\xd2\xd0\xf6\xe0\x9f\xef\xb41\xf8\
+\xc7Bn\xa0\xf2x\xa6\xf2\xf0\xe0\xbd\xc66-\xd7E\
+9\xcc<1\xde\xa6\xa2[\x94\xf2\xd9\x8b\x86\xbf\xd2\xf1\
+G\x88\xf2h\x9c\xf2\xe7\xed\xea\xa0\xbf\xf1\xed\xee\xf0\x1e\
+\xa2b\xf1\xee\xe5\xd9^T\xdd\xa3\x9f\xc6\xd6\xf1\xc99\
+/\x91\xb4M\x1c\x95Q\x0e\x95T[\xb8\x8b\xe8\xc9I\
+\x15\x8cSy\xc3\x9f\x0f\x98V\x10\x8cN\xe1\xe9\xe5\xb7\
+\xcd\xf1{\xc5\xa1\xd2\xdd\xf1\x0d\x8bN\xdf\xe6\xf1L\x9b\
+L\xf2\xdc\x92Z\x93\xf2p\xb4\x93o\xa9M\x0f\x9cX\
+R\xa8~\xd2\xc2J\xb4\xcb\xf1~\xa9\xf2\xda\xe3\xf1\xb4\
+\xb8J\x0e\x8eP\xf7\xcdJ\xbf\xd3\xb3y\xa6\xf2\xf6\xcc\
+I1\x9fS\xb3\xbdMN\xa9TT\xbb$\xa6\x00\x00\
+\x00qtRNS\x00\xb4\xf6l\xde||\xa5\xcc\xdb\
+\x09aZQ\x11\x04\xbe\x10Q\xe6e\x14j\xd6\xe6\x04\
+\x09\x02\x0c\xd5\x0b\xeaa\x0b\xb0\x1be\xb4\x17\xb4\xe1\x11\
+U\x06\xbe\xc4\x17k\x06\xe1Z\x10\x0c\xea\xd6\x22\x02\xd5\
+\xc4\x1b\xb0k\xfc\xc9\xfcUk[\xc9\x11\x11\xfdkH\
+jE\x14\xfd\xb9HkEj[\x14\xfe\xfe\xf8\xfd\xfe\
+\xfe\xfe\xfe\x07\xfe\xfe\xfe\xfeZ8\x096[\xb9\xf8\x07\
+\x096l8\x5cmZFt#+\x00\x00\x01\xd9I\
+DAT8\xcbc` \x05\xf8xz\xb0\xb2zy\
+\xbbb\x97\x95\xd1\xb7w/\x04\x03\x07]\x11.\x0ci\
+i1\xf9B$ \xc1\x8d&\xaf&P\x88\x06\xd45\
+\x91\xe5\xcdY\x0a1\x80\x995B^\x0f\x8b|a\xa1\
+\xa51L\xdeB\xa7\x10+\xb0\x81)\xb0mh\xc2\xae\
+\xc2\x08\x22obZ\xdb\x0a\x15\xe9\x9e2w\xe1\xba\xe5\
+;Wl\x9b?gr}]\xa3\xa1\x01X\x81SI\
+I\xe7L\x90tE\xf9\xa4\xb2y+7\xac^\xb3v\
+\xcf\x96\x05}-\xa5\xa5\xc5\xce`\x05.%%\xb3\x17\
+o\x02\xca\xef*\xeb)\x02\x82}\xeb\x0fl\xdd[Z\
+\x0c\x02~ \xf9\x10\xdf\x92\x92\x92\x98\xc3\x87\x0a\xcb/\
+n,\x02\x83\xed\x9bw\x17C@x\x0aP\x81?P\
+\xbe$1\xe1\xd2\xf93G\x8b\xa0`\xff\xd2\xa9P\x15\
+\xc9@\x05\x01 \x05%q\x07\x8f\x1d\x87\xc9\x17UU\
+N\x84*H\x03*`\x03+\x88\x8a?r\x1a\xae\xa0\
+\xa8\xbf\x0d\xaa\x80\x19\xae\xa0$\xf2\xc2\x0e\x84\x82\xdef\
+$\x05A\x10\x05%\xd1\xab\xe0\xf2'\x96L\x83*H\
+\x02*\x08\x84*\x88\x08[\x04S\xf0\xfa\xdc2\xa8\x82\
+T\xa0\x027;\xa8\x8a\xd0Y\xed\x10\xf9'\x0f\x1fC\
+\xe5\x99bA\x01a\x05U0\xa3k\xc2I\x90\x17>\
+V~x\x0fU\xc0\x08\x0eI\x15\xa8\x82\x92\x8e\xe9\xa7\
+\xee\xbf\xba\xfb\xe6\xce\xd3\xb7P\xf9b-\xb0\x02)a\
+\x98\x8a\x92\xb3\xd7\xee\xdd|q\xe3\x11L\x9eS\x12\x12\
+\x9d\x0a%H\xe0\xd6\x83\xdb/a\x0a\x84`\x09B\x15\
+I\xc1\xd5w\xcf`\xf2\xec\xf0$\xc7'\x8ePP}\
+\xe5:T\x9eC\x11\x91(\x95ea\xf2\xcf\xab/\xc3\
+\xe4E\x91\x93\xb5\xa0\x12TA\xcd\xe7OP\xf3\xe5\xd0\
+r\x06\x8f\x06D\xc1\x17\x88\xfby1\xb3\x9evpz\
+FI\xc9WP\xf81:\xf2c\xcf\x9eY\xb9y\xf9\
+9\x05\xd9\x99$\xe5x\x00\xfe\xa1E\xc6\x0b}\xa2(\
+\x00\x00\x00\x00IEND\xaeB`\x82\
+\x00\x00\x06v\
+\x89\
+PNG\x0d\x0a\x1a\x0a\x00\x00\x00\x0dIHDR\x00\
+\x00\x00 \x00\x00\x00 \x08\x03\x00\x00\x00D\xa4\x8a\xc6\
+\x00\x00\x00\x09pHYs\x00\x00\x0e\xc4\x00\x00\x0e\xc4\
+\x01\x95+\x0e\x1b\x00\x00\x03\x00PLTEGpL\
+C\x96\xbbs\xdeB'W\x9e#[\xa70\x81\xd51\
+\x84\xdb!U\x9f9tr%H\x8c)E\x8c*G\
+\x92`\xbb>'H\x8f.~\xda(E\x87f\xc5D\
+1\x83\xdb5\x83\xbcP\xb1\xe20WuV\xc0\xf03\
+}\xb0 J\x8a N\x94s\xdeBS\xa4Rs\xde\
+C/\x80\xd2,z\xdar\xdaD,E\x901\x84\xd8\
+\x22C\x86o\xd7K+s\xcfX\xc2\xe4,x\xdaY\
+\xb9\x95h\xce{h\xd2\x83d\xcd\x97g\xd0\x8b-y\
+\xc9Z\xbe\xad#O\x96'L\x95P\xb3\xe4$K\x92\
+p\xdaU_\xc8\xbeZ\xb0NU\xa48e\xc39$\
+O\x99d\xcd\x9e,y\xcd0\x81\xdbq\xdaG\x22Q\
+\x9at\xdeAt\xdfBq\xdbO^\xbaZo\xd9Z\
+`\xc2yk\xd4v'K\x92b\xcb\xab-y\xd1V\
+\xbf\xf3(P\x9f!J\x8a)>~t\xdeA(G\
+\x8bV\xbf\xf2'I\x8c+P\x9b*r\xc8\x22J\x8c\
+)H\x91#Q\x9d)H\x91/}\xd1'g\xa6&\
+d\xb4/|\xd1(i\xb8a\xc5~-M\x9f&c\
+\xb1,v\xce0J\xa26\x85\xba&U\xa2^\xc7\xc4\
+!\x5c\xa2V\xbf\xf1?\x90\xa1c\xca\x8e+t\xc1+\
+u\xc1\x5c\xc3\xb9E\x9b\xafV\xbf\xef^\xc3\x9c.|\
+\xdaF\xa1\xcb1j\xb7=\x90\xba-v\xb8V\xbf\xf2\
+K\xa6\xba<\x8e\xc1m\xd6jU\xbb\xdcK\xa8\xd3W\
+\xc0\xed.w\xbaX\xc1\xe71}\xbb,x\xd4W\xc0\
+\xeb>\x93\xc6V\xbc\xe0Q\xb4\xd9 P\x960{\xba\
+T\xb4\xbc$a\xa0n\xd7cY\xc1\xe3%h\xbaS\
+\xb8\xd9[\xc4\xd47\x87\xbfq\xdbM-/qp\xda\
+ST\xbb\xe9Z\xc3\xd83\x86\xdaW\xc0\xecW\xc0\xe9\
+Y\xc2\xdcP\xae\xc3L\xab\xcc;\x8e\xc00~\xc74\
+\x82\xbf,y\xda%Z\xa9(m\xc4!S\x9a'g\
+\xbe'h\xbe+x\xda*r\xd1+x\xda+w\xd9\
+2\x86\xdb3\x88\xdb0\x83\xdb3\x87\xdb\x1fK\x8c2\
+\x85\xdb\x1fN\x93/\x80\xdb\x1fQ\x99t\xdfA L\
+\x8f M\x91\x1fI\x8a\x1fO\x95q\xdbPX\xc1\xe5\
+ J\x8b1\x84\xdb\x1fP\x97.\x7f\xda R\x9an\
+\xd8dq\xdcLo\xd9\x5cp\xdaV-|\xdar\xdc\
+J\x1fK\x8dY\xc2\xe1Y\xc3\xddk\xd5tl\xd6n\
+g\xd1\x8aa\xcb\xb0b\xcc\xabd\xce\x9d_\xc9\xb8i\
+\xd3\x82h\xd2\x86m\xd6jc\xcd\xa5`\xca\xb4s\xdd\
+Cp\xd9Y3\x88\xdap\xdaSs\xddFX\xc1\xe9\
+/}\xc8,z\xda3\x87\xdak\xd5rj\xd3|j\
+\xd4xi\xd3~W\xc0\xeff\xd0\x91g\xd0\x8e_\xc8\
+\xbcf\xcf\x94m\xd7ge\xcf\x96c\xcd\xa1s\xdcG\
+^\xc7\xc1n\xd8an\xd8_W\xc0\xea]\xc7\xc7\x5c\
+\xc6\xcc]\xc6\xc9W\xc0\xec\x5c\xc5\xd1\x5c\xc5\xce[\xc4\
+\xd53\x86\xd5Z\xc3\xda[\xc4\xd3/~\xceZ\xc4\xd7\
+1\x83\xd12\x85\xd8e\xce\x99s\xddD R\x9ca\
+\xcb\xa7_\xc8\xbf0\x80\xcd,w\xc4^\xc7\xc4/|\
+\xc5-x\xc01\x83\xd53\x88\xd9.{\xc2|7\x1f\
+\xe1\x00\x00\x00\xa1tRNS\x00\x16\xc1\x1a\xfd\xfe\xfe\
+\xfd\x04\xc12\x8d3\x80\xf8I.\xb7\xfd7\x09\xfd\xfd\
+\xfe\xfc\xcd\x0d\xa8g\xa0\xd9g\xbe\x0fX\xfd\xc9\xfe%\
+H\xfcXhG;\xd8\x9f?\xe3\xf7\xa5B\x1b$\xcc\
+\xf4B\xect\xe7\xb0\xec\xf4P\x84\x91\xe7\xae\xb8N\xac\
+\x91\xf5)\x80_\xe4v[:\xf4\x9a\xdf\x87\xfdF\xfd\
+\xfd\xfc\xf7o\xfc\xfdh\xfd\xbb\xf3\x12\xf2\xf9\xfd\xa2`\
+\xfd\xfc\xec\xfd\xe3\xfdm\xfd\xfd\xb6\xfd\xfd\x89\xfd#\xcd\
+\xfe\xf3\xfd\xfdz\xfc\xfe\xfd\xfd\xfdC\x1f\xf0\x86.\xe8\
+y\xfc\xca\x1f\x9b_\xa2\x9d\x93\xb6\xbdG\x8a\xbc\xcf\xd5\
+\xf5\x99Q\xfd\xba\xfc\xb2\xc9\x8b\xc8u|\x80\xe1\x00\x00\
+\x02oIDAT8\xcbc` \x010\xf6\xf4O\
+PQ\x99\xd8\xdb\xde(\x82MZ\xbf@\xf4\xc5\x85\x0d\
+{\xf7}x\xf7\xeay\x82\x91\x0e\x86n7\xd1\x1bO\
+\x80\xd2o_\xbf|\xf6\xf4\xe1\xed\xa3\x87\x5c\xd4P\xe4\
+\x85}n\xc0\xb5\xff\xfa~\xf4\xd4\xa1\xc3'\xcd5\x91\
+\xe5\x93P\xb5\x1f>y\xff\xc8\xbd\x9bZp\xf9\xd2L\
+t\xedG>\xdf\xb9uPC\x1d*\xcfQ\x86\xac\xfd\
+\xd4\xe1o \xed\x07\x8f\x1f\xbb\xea$\x05QP}a\
+CIN\x91\x90P\x5c^a:H\xfb}\x90\xf6c\
+\xd7\xaf\xed?P\x0c\xf1@EmM\xf9\xff\x8b?~\
+\xfc\xb8\xf8'?\x03\xae\xfd\xda\x95\x03w\x9b\xc1F\xb4\
+\xd4U\xfe\xfe\xf8\xe6\xdf\xbf\xb3\xff>\xfd}\xffS,\
+\x0b\xa8\xfd8H\xfb\x89m\x8fv(\x81\x5c\xd0\xd6p\
+\xf1\xcd\xd9E@\xb0d\xc9\xc2O\xac?\xc5R\xa0\xda\
+\xb7=>\xb3\xb3\x1b\xa8`R\xc7o\xb8\xfcBA\x99\
+\xb4\xd4d\xb0\xf6\xbb\x8fv\x9c9\xb7\xbe\x13\xa8`\xf2\
+\x9f\x8fg\xa1\xd2\xac\xb9 +C#\xc1\xdaw\xec<\
+\xb7~\xb7\x1c\x90?\xe5\xe2'\xa8|H<\xc4W\x8e\
+`\xed;\xd7o\xdf}^\x12\xc8\x9d\xf6\xe3\x1fXz\
+\x19+{\x14D\x81=D\xfb\xf6\x07_NK\x03\xb9\
+S?^\x06\xcb/\x8b\xde\x1a\x0bQ`\x0c\xd1\xfe\xe0\
+\xcb\xaaU _(|\x02Ko\x0a_\xf9\xd5\x1d\xa2\
+\xc0\x01\xa4}\xf7\xf9\xd3\xabV\xc9q\x00\xb9}\x0b\xc1\
+\xf2\x8b#V~\x0d\x02\xf1\x19\x04\x0c\xc1\xda\x81\xf2\xb6\
+\x02 \xbe\x0cHz\xd3\xe2\xc5,[\xb7n\xf5\x03\x09\
+\xf0\xd8A\xb4\xafb2\x05\x1b\xd8\xc4\x06\xd4\x0e\x04\xec\
++\xb7n\xb5f\x06\x89\x98XI3I\xdax\xf1\xc0\
+\x22[\x01,\xbf\x02\xa8`\xe5J3E,\x89Q\x9b\
+\x0d(\xbdb\x05\xcb\xe6\x95+7o\xd6\xe5\xc2\xa2B\
+\x10$\xbf%l\xdd\xe6\xcd\x9b\xd7\xad\x93\xd0\x83)\x11\
+\x91\xe7\x86\xb2\x98-\x81\xf2[b\xd6\xac\x03\x82\xe5\xcb\
+\xd7\xac1p\xe6\xe6\x0d\x0e\xe4\x0c\x10\xe7\x87\x19a\xc1\
+\xb7e\xcb\x96],k\x96/\x07\xc9\xafY\xbdz\xcf\
+\x9e\xa5\x1b7\x8a\xfb\x22,q\xe5\xdb\xb5kWU\xfd\
+\x9a5\xcbA\xd2\xab\xf7,\x05\xc9\xf3\x22;\xc3?\x1b\
+\xa8BYb\xcf\xea\xd5P\xed\x1b=\xbcQ\x1d\x9a(\
+\xbb\xeb\xd2%e\xf6\x8dKA\x00\xa8\x9d\xd3\x13\xc3/\
+\xad\xb2\xd3UU\xe7\xcc\x12_\xbbv\xad8'?\xd6\
+\xcc\xab8s\xfe\xdc\x05\xf3f\xcf\x90\xef\x22%\xcb\x03\
+\x00/\xab\x8c\xa71\xf3a\xb4\x00\x00\x00\x00IEN\
+D\xaeB`\x82\
+"
+
+qt_resource_name = b"\
+\x00\x06\
+\x07\x03}\xc3\
+\x00i\
+\x00m\x00a\x00g\x00e\x00s\
+\x00\x0d\
+\x06\x0b\xf4\xa7\
+\x00b\
+\x00l\x00a\x00n\x00k\x00_\x001\x002\x008\x00.\x00p\x00n\x00g\
+\x00\x12\
+\x0d,\x1a\xc7\
+\x00d\
+\x00a\x00i\x00l\x00y\x00c\x00h\x00e\x00c\x00k\x00_\x001\x002\x008\x00.\x00p\x00n\
+\x00g\
+\x00\x08\
+\x09n\x91\xb3\
+\x00b\
+\x00r\x00o\x00w\x00s\x00e\x00r\x00s\
+\x00\x0c\
+\x0fTN'\
+\x00b\
+\x00r\x00a\x00v\x00e\x00_\x003\x002\x00.\x00p\x00n\x00g\
+\x00\x0d\
+\x01\x97\x91\x87\
+\x00c\
+\x00h\x00r\x00o\x00m\x00e\x00_\x003\x002\x00.\x00p\x00n\x00g\
+\x00\x0b\
+\x02\xb2\x0c'\
+\x00e\
+\x00d\x00g\x00e\x00_\x003\x002\x00.\x00p\x00n\x00g\
+"
+
+qt_resource_struct = b"\
+\x00\x00\x00\x00\x00\x02\x00\x00\x00\x01\x00\x00\x00\x01\
+\x00\x00\x00\x00\x00\x00\x00\x00\
+\x00\x00\x00\x00\x00\x02\x00\x00\x00\x03\x00\x00\x00\x02\
+\x00\x00\x00\x00\x00\x00\x00\x00\
+\x00\x00\x00\x12\x00\x00\x00\x00\x00\x01\x00\x00\x00\x00\
+\x00\x00\x01\x8db\xb8\xc9\xc8\
+\x00\x00\x00\x5c\x00\x02\x00\x00\x00\x03\x00\x00\x00\x05\
+\x00\x00\x00\x00\x00\x00\x00\x00\
+\x00\x00\x002\x00\x00\x00\x00\x00\x01\x00\x00\x01\xbc\
+\x00\x00\x01\x8df\xb35\xbe\
+\x00\x00\x00\x90\x00\x00\x00\x00\x00\x01\x00\x00\x15\xae\
+\x00\x00\x01\x8dd\xf15i\
+\x00\x00\x00\xb0\x00\x00\x00\x00\x00\x01\x00\x00\x1b/\
+\x00\x00\x01\x8dd\xf1K\xbc\
+\x00\x00\x00r\x00\x00\x00\x00\x00\x01\x00\x00\x10,\
+\x00\x00\x01\x8dd\xf1\x5c\xbe\
+"
+
+def qInitResources():
+ QtCore.qRegisterResourceData(0x03, qt_resource_struct, qt_resource_name, qt_resource_data)
+
+def qCleanupResources():
+ QtCore.qUnregisterResourceData(0x03, qt_resource_struct, qt_resource_name, qt_resource_data)
+
+qInitResources()
diff --git a/dailycheck.icns b/dailycheck.icns
new file mode 100644
index 0000000..b4fdf33
Binary files /dev/null and b/dailycheck.icns differ
diff --git a/dailycheck.ico b/dailycheck.ico
new file mode 100644
index 0000000..4ab6b0b
Binary files /dev/null and b/dailycheck.ico differ
diff --git a/global_vars.py b/global_vars.py
new file mode 100644
index 0000000..789bd94
--- /dev/null
+++ b/global_vars.py
@@ -0,0 +1,42 @@
+# coding: utf8
+import requests
+from PySide6 import QtWidgets
+
+SoftwareStatusRole = 0x0101
+ExtensionStatusRole = 0x0102
+ExtensionIdRole = 0x0103
+
+
+def accept_warning(widget: QtWidgets.QWidget, condition: bool,
+ caption: str = "Warning", text: str = "Are you sure to continue?") -> bool:
+ if condition:
+ b = QtWidgets.QMessageBox.question(widget, caption, text)
+ if b == QtWidgets.QMessageBox.StandardButton.No:
+ return True
+ return False
+
+
+def get_with_chained_keys(dic: dict, keys: list, default=None):
+ """
+ 调用 get_with_chained_keys(d, ["a", "b", "c"])
+ 等同于 d["a"]["b"]["c"] ,
+ 只不过中间任意一次索引如果找不到键,则返回 default
+
+ :param dic: 目标字典
+ :param keys: 键列表
+ :param default: 找不到键时的默认返回值
+ :return:
+ """
+ k = keys[0]
+ if k not in dic:
+ return default
+ if len(keys) == 1:
+ return dic[k]
+ return get_with_chained_keys(dic[k], keys[1:], default)
+
+
+def request_content(url: str) -> bytes:
+ req = requests.get(url)
+ if req.status_code == 200:
+ return req.content
+ return b""
diff --git a/images/blank_128.png b/images/blank_128.png
new file mode 100644
index 0000000..7eeafdb
Binary files /dev/null and b/images/blank_128.png differ
diff --git a/images/browsers/brave_32.png b/images/browsers/brave_32.png
new file mode 100644
index 0000000..551243e
Binary files /dev/null and b/images/browsers/brave_32.png differ
diff --git a/images/browsers/chrome_32.png b/images/browsers/chrome_32.png
new file mode 100644
index 0000000..f00d8a8
Binary files /dev/null and b/images/browsers/chrome_32.png differ
diff --git a/images/browsers/edge_32.png b/images/browsers/edge_32.png
new file mode 100644
index 0000000..f8507c4
Binary files /dev/null and b/images/browsers/edge_32.png differ
diff --git a/images/dailycheck_128.png b/images/dailycheck_128.png
new file mode 100644
index 0000000..059eeb6
Binary files /dev/null and b/images/dailycheck_128.png differ
diff --git a/main.py b/main.py
new file mode 100644
index 0000000..8483b59
--- /dev/null
+++ b/main.py
@@ -0,0 +1,74 @@
+# coding: utf8
+import os
+import sys
+from pathlib import Path
+from PySide6 import QtWidgets, QtCore
+from mw_dailycheck import MwDailyCheck
+
+import daily_check_rc
+
+version = (1, 0, 0)
+
+ORG_NAME = "JnPrograms"
+APP_NAME = "DailyCheck"
+
+
+def set_default_settings():
+ plat = sys.platform
+ user_path = os.path.expanduser("~")
+ user_data_path_map = {
+ "win32": {
+ "Chrome": str(Path(user_path, r"AppData\Local\Google\Chrome\User Data")),
+ "Edge": str(Path(user_path, r"AppData\Local\Microsoft\Edge\User Data")),
+ "Brave": str(Path(user_path, r"AppData\Local\BraveSoftware\Brave-Browser\User Data")),
+ },
+ "darwin": {
+ "Chrome": str(Path(user_path, "Library/Application Support/Google/Chrome")),
+ "Edge": str(Path(user_path, "Library/Application Support/Microsoft Edge")),
+ "Brave": str(Path(user_path, "Library/Application Support/BraveSoftware/Brave-Browser")),
+ },
+ }
+ exec_path_map = {
+ "win32": {
+ "Chrome": r"C:\Program Files\Google\Chrome\Application\chrome.exe",
+ "Edge": r"C:\Program Files (x86)\Microsoft\Edge\Application\msedge.exe",
+ "Brave": r"C:\Program Files\BraveSoftware\Brave-Browser\Application\brave.exe",
+ },
+ "darwin": {
+ "Chrome": r"/Applications/Google Chrome.app/Contents/MacOS/Google Chrome",
+ "Edge": r"/Applications/Microsoft Edge.app/Contents/MacOS/Microsoft Edge",
+ "Brave": r"/Applications/Brave Browser.app/Contents/MacOS/Brave Browser",
+ },
+ }
+ user_data_path = user_data_path_map[plat]
+ exec_path = exec_path_map[plat]
+ settings_map = {
+ "ChromeExec": exec_path["Chrome"],
+ "EdgeExec": exec_path["Edge"],
+ "BraveExec": exec_path["Brave"],
+ "ChromeData": user_data_path["Chrome"],
+ "EdgeData": user_data_path["Edge"],
+ "BraveData": user_data_path["Brave"],
+ }
+ us = QtCore.QSettings()
+ exist_keys = us.childKeys()
+ for s in settings_map:
+ if s not in exist_keys:
+ us.setValue(s, settings_map[s])
+ # print(f"Add key {s}")
+
+
+def main():
+ app = QtWidgets.QApplication(sys.argv)
+ app.setOrganizationName(ORG_NAME)
+ app.setApplicationName(APP_NAME)
+
+ set_default_settings()
+
+ win = MwDailyCheck(version)
+ win.show()
+ return app.exec()
+
+
+if __name__ == '__main__':
+ main()
diff --git a/mw_dailycheck.py b/mw_dailycheck.py
new file mode 100644
index 0000000..daaad78
--- /dev/null
+++ b/mw_dailycheck.py
@@ -0,0 +1,101 @@
+# coding: utf8
+import json
+from pathlib import Path
+from datetime import datetime
+from PySide6 import QtWidgets, QtGui, QtCore
+from wg_basic import WgBasic
+from wg_software import WgSoftware
+from wg_extensions import WgExtensions
+from global_vars import (
+ request_content,
+ accept_warning,
+)
+
+
+class UiMwDailyCheck(object):
+
+ def __init__(self, window: QtWidgets.QMainWindow):
+
+ self.cw = QtWidgets.QWidget(window)
+ window.setCentralWidget(self.cw)
+
+ self.vly_m = QtWidgets.QVBoxLayout()
+ self.cw.setLayout(self.vly_m)
+ self.tw_m = QtWidgets.QTabWidget(self.cw)
+ self.vly_m.addWidget(self.tw_m)
+ self.wg_basic = WgBasic(self.cw)
+ self.wg_software = WgSoftware(self.cw)
+ self.wg_extensions = WgExtensions(self.cw)
+ self.tw_m.addTab(self.wg_basic, "基本信息")
+ self.tw_m.addTab(self.wg_software, "已安装软件")
+ self.tw_m.addTab(self.wg_extensions, "已安装插件")
+
+ self.menu_bar = window.menuBar()
+ self.menu_help = self.menu_bar.addMenu("帮助")
+ self.menu_about = self.menu_bar.addMenu("关于")
+
+ self.act_update_safe = QtGui.QAction("更新安全标注", window)
+ self.act_export_unknown = QtGui.QAction("导出未知", window)
+
+ self.act_about = QtGui.QAction("关于", window)
+ self.act_about_qt = QtGui.QAction("关于 Qt", window)
+
+ self.menu_help.addActions([self.act_update_safe, self.act_export_unknown])
+ self.menu_about.addActions([self.act_about, self.act_about_qt])
+
+
+class MwDailyCheck(QtWidgets.QMainWindow):
+
+ def __init__(self, version: tuple, parent=None):
+ super().__init__(parent)
+ self.version = version
+ self.setWindowTitle("日常检查工具")
+ self.setWindowIcon(QtGui.QIcon(":/images/dailycheck_128.png"))
+ self.ui = UiMwDailyCheck(self)
+
+ self.ui.act_update_safe.triggered.connect(self.on_act_update_safe_triggered)
+ self.ui.act_export_unknown.triggered.connect(self.on_act_export_unknown_triggered)
+ self.ui.act_about.triggered.connect(self.on_act_about_triggered)
+ self.ui.act_about_qt.triggered.connect(self.on_act_about_qt_triggered)
+
+ def sizeHint(self):
+ return QtCore.QSize(800, 600)
+
+ def on_act_about_triggered(self):
+ msg = f"日常检查工具\n\n适用于 Windows 平台。版本:{self.version[0]}.{self.version[1]}.{self.version[-1]}"
+ QtWidgets.QMessageBox.about(self, "关于", msg)
+
+ def on_act_about_qt_triggered(self):
+ QtWidgets.QMessageBox.aboutQt(self, "关于 Qt")
+
+ def on_act_update_safe_triggered(self):
+ root_url = "https://julianfreeman.github.io/dailycheckutils"
+ marks_all = request_content(f"{root_url}/marks_all.json").decode("utf8")
+ if len(marks_all) == 0:
+ return
+ marks_all_d = json.loads(marks_all)
+ self.ui.wg_software.update_safe(marks_all_d["software_win"])
+ self.ui.wg_extensions.update_safe(marks_all_d["extensions"])
+ self.ui.wg_basic.update_safe(marks_all_d["isp"], marks_all_d["manufacturer"])
+
+ def on_act_export_unknown_triggered(self):
+ dirname = QtWidgets.QFileDialog.getExistingDirectory(self, "导出未知")
+ if len(dirname) == 0:
+ return
+ now = datetime.strftime(datetime.now(), "%y%m%d%H%M")
+ ex_file = Path(dirname, f"未知信息{now}.json")
+ if accept_warning(self, ex_file.exists(), "警告", "文件已存在,确认覆盖吗?"):
+ return
+
+ unknown_software = self.ui.wg_software.export_unknown()
+ unknown_ext = self.ui.wg_extensions.export_unknown()
+ unknown_isp_manu = self.ui.wg_basic.export_unknown()
+
+ unknown_all = {
+ "software_win": unknown_software,
+ "extensions": unknown_ext,
+ }
+ unknown_all.update(unknown_isp_manu)
+
+ ex_file.write_text(json.dumps(unknown_all, indent=4, ensure_ascii=False), "utf8")
+ QtWidgets.QMessageBox.information(self, "提示", f"已导出到 {ex_file}")
diff --git a/util_ext.py b/util_ext.py
new file mode 100644
index 0000000..4f2f946
--- /dev/null
+++ b/util_ext.py
@@ -0,0 +1,237 @@
+# coding: utf8
+import os
+import json
+import shutil
+from pathlib import Path
+# from typing import Callable
+from PySide6 import QtCore, QtWidgets
+from dataclasses import dataclass, field
+from global_vars import get_with_chained_keys
+
+
+@dataclass
+class ProfileNode(object):
+ gaia_given_name: str
+ gaia_name: str
+ name: str
+ shortcut_name: str
+ user_name: str
+
+
+ProfilesData = dict[str, ProfileNode]
+
+
+def scan_profiles(user_data_path: str) -> ProfilesData:
+ local_state_path = Path(user_data_path, "Local State")
+ if not local_state_path.exists():
+ return {}
+
+ local_state_data: dict = json.loads(local_state_path.read_text(encoding="utf8"))
+ info_cache_data: dict = get_with_chained_keys(local_state_data, ["profile", "info_cache"])
+
+ profiles_data: ProfilesData = {}
+
+ for profile_id in info_cache_data:
+ profile_data: dict = info_cache_data[profile_id]
+ profile_node = ProfileNode(
+ gaia_given_name=profile_data.get("gaia_given_name", ""),
+ gaia_name=profile_data.get("gaia_name", ""),
+ name=profile_data.get("name", ""),
+ shortcut_name=profile_data.get("shortcut_name", ""),
+ user_name=profile_data.get("user_name"),
+ )
+ profiles_data[profile_id] = profile_node
+
+ return profiles_data
+
+
+@dataclass
+class ExtensionNode(object):
+ # ids: str
+ icon: str
+ name: str
+ # path: str
+ profiles: list[str] = field(default_factory=list)
+
+
+ExtensionsData = dict[str, ExtensionNode]
+
+
+def get_extension_icon_path(ext_icons: dict[str, str], ext_path: str, profile_path: Path) -> str:
+ if len(ext_icons) == 0:
+ return ""
+
+ if "128" in ext_icons:
+ icon_file = ext_icons["128"]
+ else:
+ icon_file = ext_icons[str(max(map(lambda x: int(x), ext_icons.keys())))]
+ # 如果路径以 / 开头,会被认为是根而忽略其他,因此需要检查一下
+ if icon_file.startswith("/"):
+ icon_file = icon_file[1:]
+
+ full_path = Path(profile_path, "Extensions", ext_path, icon_file)
+ if not full_path.exists():
+ return ""
+ return str(full_path)
+
+
+def scan_extensions(browser: str, is_compat=False) -> tuple[ExtensionsData, ProfilesData]:
+ us = QtCore.QSettings()
+ user_data_path = str(us.value(f"{browser}Data", ""))
+ if len(user_data_path) == 0 or not Path(user_data_path).exists():
+ return {}, {}
+
+ profile_data = scan_profiles(user_data_path)
+ extensions_data: ExtensionsData = {}
+
+ if is_compat:
+ pref_file = "Preferences"
+ else:
+ pref_file = "Secure Preferences"
+
+ # print(pref_file)
+
+ for profile_id in profile_data:
+ profile_path = Path(user_data_path, profile_id)
+ secure_pref_path = Path(profile_path, pref_file)
+ secure_pref_data: dict = json.loads(secure_pref_path.read_text(encoding="utf8"))
+ ext_settings_data: dict = get_with_chained_keys(secure_pref_data, ["extensions", "settings"], dict())
+
+ for ext_id in ext_settings_data:
+ ext_data: dict = ext_settings_data[ext_id]
+ # 这里插件有几种情况
+ ext_path: str = ext_data.get("path", "")
+ if len(ext_path) == 0:
+ # 如果 path 是空,则该插件可能是一些内部插件,不予收集
+ continue
+ elif not (ext_path.startswith(ext_id) or os.path.exists(ext_path)):
+ # 如果 path 以插件 ID 开头,则为商店安装的插件,
+ # 如果不是以插件 ID 开头,但是是一个存在的路径,则为离线安装的插件
+ # 否则,可能也是内部插件,不予收集
+ continue
+ ext_manifest: dict = ext_data.get("manifest", {})
+ if len(ext_manifest) == 0:
+ # 如果 manifest 为空,则该插件可能是离线插件,需要额外找它的 manifest
+ ext_manifest_path = Path(ext_path, "manifest.json")
+ ext_manifest: dict = json.loads(ext_manifest_path.read_text(encoding="utf8"))
+
+ if ext_id not in extensions_data:
+ ext_node = ExtensionNode(
+ # ids=ext_id,
+ icon=get_extension_icon_path(ext_manifest.get("icons", {}), ext_path, profile_path),
+ name=ext_manifest.get("name", ""),
+ # path=ext_data.get("path", ""),
+ profiles=[profile_id],
+ )
+ extensions_data[ext_id] = ext_node
+ else:
+ ext_node = extensions_data[ext_id]
+ ext_node.profiles += [profile_id]
+
+ return extensions_data, profile_data
+
+
+# ================== Deletion ====================
+
+
+def delete_extensions(profile_path: str, pref_name: str, ext_ids: list[str]) -> tuple[int, int]:
+ total = len(ext_ids)
+
+ e_pref_path = Path(profile_path, pref_name)
+ e_pref_data = json.loads(e_pref_path.read_text("utf8")) # type: dict
+ ext_set_data = get_with_chained_keys(e_pref_data, ["extensions", "settings"]) # type: dict
+ if ext_set_data is None:
+ return 0, total
+
+ s_pref_path = Path(profile_path, "Secure Preferences")
+ pref_path = Path(profile_path, "Preferences")
+ s_pref_data = json.loads(s_pref_path.read_text("utf8")) # type: dict
+ pref_data = json.loads(pref_path.read_text("utf8")) # type: dict
+
+ macs = get_with_chained_keys(s_pref_data, ["protection", "macs", "extensions", "settings"]) # type: dict
+ if macs is None:
+ return 0, total
+
+ success = 0
+ for ids in ext_ids:
+ c1 = ext_set_data.pop(ids, None)
+ c2 = macs.pop(ids, None)
+ if None not in (c1, c2):
+ success += 1
+
+ pinned_ext = get_with_chained_keys(pref_data, ["extensions", "pinned_extensions"]) # type: list
+ if pinned_ext is not None:
+ for ids in ext_ids:
+ if ids in pinned_ext:
+ pinned_ext.remove(ids)
+
+ s_pref_path.write_text(json.dumps(s_pref_data, ensure_ascii=False), "utf8")
+ pref_path.write_text(json.dumps(pref_data, ensure_ascii=False), "utf8")
+
+ extensions_path_d = Path(profile_path, "Extensions")
+ for ids in ext_ids:
+ # 对于离线安装的插件,目录可能不在这个位置,所以就不删了
+ ext_folder_path = Path(extensions_path_d, ids)
+ if ext_folder_path.exists():
+ shutil.rmtree(ext_folder_path, ignore_errors=True)
+
+ return success, total
+
+
+class DeleteThread(QtCore.QThread):
+
+ deleted = QtCore.Signal(int, int)
+
+ def __init__(self,
+ profile_path: str,
+ pref_name: str,
+ ext_ids: list[str],
+ parent: QtCore.QObject = None):
+ super().__init__(parent)
+ self.profile_path = profile_path
+ self.pref_name = pref_name
+ self.ext_ids = ext_ids
+ self.finished.connect(self.deleteLater)
+
+ def run(self):
+ success, total = delete_extensions(self.profile_path, self.pref_name, self.ext_ids)
+ self.deleted.emit(success, total)
+
+
+class DeleteThreadManager(QtCore.QObject):
+
+ def __init__(self, total: int, progress_bar: QtWidgets.QProgressBar, parent: QtWidgets.QDialog):
+ super().__init__(parent)
+ self.deletion_progress = 0
+ self.success_deletion = 0
+ self.fail_deletion = 0
+ self.total_for_deletion = total
+ self.deletion_info = "成功:{success} 个;失败:{fail} 个;总共 {total} 个。"
+ self.progress_bar = progress_bar
+ self.parent = parent
+
+ self.progress_bar.setMaximum(total)
+ self.progress_bar.setValue(0)
+
+ self.progress_bar.valueChanged.connect(self.on_pgb_del_value_changed)
+
+ def start(self, thread: DeleteThread):
+ thread.deleted.connect(self.on_del_thd_deleted)
+ thread.start()
+
+ def on_del_thd_deleted(self, success: int, total: int):
+ self.success_deletion += success
+ self.deletion_progress += total
+ self.fail_deletion += total - success
+ self.progress_bar.setValue(self.deletion_progress)
+
+ def on_pgb_del_value_changed(self, value: int):
+ if value == self.total_for_deletion:
+ QtWidgets.QMessageBox.information(
+ self.parent, "删除结果", self.deletion_info.format(
+ success=self.success_deletion,
+ fail=self.fail_deletion,
+ total=self.total_for_deletion
+ )
+ )
+ self.parent.accept()
diff --git a/util_func.py b/util_func.py
new file mode 100644
index 0000000..915ad30
--- /dev/null
+++ b/util_func.py
@@ -0,0 +1,82 @@
+# coding: utf8
+import os
+import wmi
+import json
+import win32gui
+import subprocess
+from pathlib import Path
+from PySide6.QtCore import QDir, QSettings
+from PySide6.QtGui import QImage, QIcon, QPixmap
+from global_vars import request_content
+
+
+def get_isp_name() -> str:
+ try:
+ data = json.loads(request_content("https://ipinfo.io/"))
+ return data.get("org", "[Not found]")
+ except json.JSONDecodeError:
+ return "[Decode Error]"
+
+
+def get_win_manufacturer() -> list[str]:
+ w = wmi.WMI()
+ cs_ls = w.Win32_ComputerSystem()
+ if len(cs_ls) > 0:
+ cs = cs_ls[0]
+ return [cs.model, cs.Manufacturer]
+ return []
+
+
+def get_win_license() -> str:
+ temp_out = str(Path(QDir.tempPath(), "lic.txt"))
+ return_code = subprocess.call(
+ rf'cscript /Nologo /U "C:\Windows\System32\slmgr.vbs" /dlv >{temp_out}',
+ shell=True,
+ )
+ if return_code == 0:
+ with open(temp_out, "r", encoding="utf-16-le") as f:
+ return f.read()
+ return ""
+
+
+def get_win_installed_software() -> dict[str, str]:
+
+ def gather_software(path: str, software_dict: dict[str, str]):
+ reg = QSettings(path, QSettings.Format.NativeFormat)
+ soft = reg.childGroups()
+ for s in soft:
+ c = QSettings(path + "\\" + s.split("/")[0], QSettings.Format.NativeFormat)
+ name = str(c.value("DisplayName"))
+ if name == "None":
+ continue
+ icon = str(c.value("DisplayIcon"))
+ software_dict[name] = icon
+
+ all_software = {}
+ gather_software(r"HKEY_LOCAL_MACHINE\SOFTWARE\Microsoft\Windows\CurrentVersion\Uninstall", all_software)
+ gather_software(r"HKEY_LOCAL_MACHINE\SOFTWARE\WOW6432Node\Microsoft\Windows\CurrentVersion\Uninstall", all_software)
+ gather_software(r"HKEY_CURRENT_USER\Software\Microsoft\Windows\CurrentVersion\Uninstall", all_software)
+
+ return all_software
+
+
+def extract_win_icon_from_file(icon: str, default: QIcon) -> QIcon:
+ if icon == "None":
+ return default
+ if icon.endswith(",0"):
+ icon = icon[:-2]
+ icon = icon.replace('"', "")
+ icon = icon.replace("'", "")
+ if not os.path.exists(icon):
+ return default
+ if not icon.endswith(("exe", "EXE", "dll", "DLL")):
+ return QIcon(icon)
+
+ large, small = win32gui.ExtractIconEx(icon, 0)
+ if len(small) > 0:
+ win32gui.DestroyIcon(small[0])
+ if len(large) > 0:
+ image = QImage.fromHICON(large[0])
+ return QIcon(QPixmap(image))
+ else:
+ return default
diff --git a/wg_basic.py b/wg_basic.py
new file mode 100644
index 0000000..cac38dc
--- /dev/null
+++ b/wg_basic.py
@@ -0,0 +1,92 @@
+# coding: utf8
+from PySide6 import QtWidgets, QtCore, QtGui
+from util_func import (
+ get_isp_name,
+ get_win_manufacturer,
+ get_win_license,
+)
+
+
+class UiWgBasic(object):
+
+ def __init__(self, window: QtWidgets.QWidget):
+ self.vly_m = QtWidgets.QVBoxLayout()
+ window.setLayout(self.vly_m)
+
+ self.gbx_isp = QtWidgets.QGroupBox("网络运营商", window)
+ self.vly_m.addWidget(self.gbx_isp)
+ self.vly_gbx_isp = QtWidgets.QVBoxLayout()
+ self.gbx_isp.setLayout(self.vly_gbx_isp)
+
+ self.lne_isp = QtWidgets.QLineEdit(self.gbx_isp)
+ self.lne_isp.setReadOnly(True)
+ self.vly_gbx_isp.addWidget(self.lne_isp)
+
+ self.gbx_manu = QtWidgets.QGroupBox("系统制造商", window)
+ self.vly_m.addWidget(self.gbx_manu)
+ self.hly_gbx_manu = QtWidgets.QHBoxLayout()
+ self.gbx_manu.setLayout(self.hly_gbx_manu)
+ self.lb_model = QtWidgets.QLabel("型号:", self.gbx_manu)
+ self.lne_model = QtWidgets.QLineEdit(self.gbx_manu)
+ self.lne_model.setReadOnly(True)
+ self.lb_manufacturer = QtWidgets.QLabel("制造商:", self.gbx_manu)
+ self.lne_manufacturer = QtWidgets.QLineEdit(self.gbx_manu)
+ self.lne_manufacturer.setReadOnly(True)
+ self.hly_gbx_manu.addWidget(self.lb_model)
+ self.hly_gbx_manu.addWidget(self.lne_model)
+ self.hly_gbx_manu.addWidget(self.lb_manufacturer)
+ self.hly_gbx_manu.addWidget(self.lne_manufacturer)
+
+ self.gbx_license = QtWidgets.QGroupBox("许可证", window)
+ self.vly_m.addWidget(self.gbx_license)
+ self.vly_gbx_license = QtWidgets.QVBoxLayout()
+ self.gbx_license.setLayout(self.vly_gbx_license)
+ self.pte_license = QtWidgets.QPlainTextEdit(self.gbx_license)
+ self.vly_gbx_license.addWidget(self.pte_license)
+ self.pte_license.setReadOnly(True)
+
+
+class WgBasic(QtWidgets.QWidget):
+
+ def __init__(self, parent=None):
+ super().__init__(parent)
+ self.ui = UiWgBasic(self)
+
+ self.ui.lne_isp.setText(get_isp_name())
+ model, manufacturer = get_win_manufacturer()
+ self.ui.lne_model.setText(model)
+ self.ui.lne_manufacturer.setText(manufacturer)
+
+ self.ui.pte_license.setPlainText(get_win_license())
+
+ def update_safe(self, isp_safe_info: dict, manu_safe_info: dict):
+
+ def set_palette(is_safe: bool | None, lne_w: QtWidgets.QLineEdit):
+ pal = lne_w.palette()
+ if is_safe is True:
+ pal.setColor(QtGui.QPalette.ColorRole.Text, QtCore.Qt.GlobalColor.blue)
+ elif is_safe is False:
+ pal.setColor(QtGui.QPalette.ColorRole.Text, QtCore.Qt.GlobalColor.red)
+ else:
+ pal.setColor(QtGui.QPalette.ColorRole.Text, QtCore.Qt.GlobalColor.black)
+ lne_w.setPalette(pal)
+
+ isp_text = self.ui.lne_isp.text()
+ if isp_text in isp_safe_info:
+ is_isp_safe = isp_safe_info[isp_text]["safe"]
+ set_palette(is_isp_safe, self.ui.lne_isp)
+
+ manu_text = self.ui.lne_manufacturer.text()
+ if manu_text in manu_safe_info:
+ is_manu_safe = manu_safe_info[manu_text]["safe"]
+ set_palette(is_manu_safe, self.ui.lne_manufacturer)
+
+ def export_unknown(self) -> dict:
+ unknown = {}
+ text_role = QtGui.QPalette.ColorRole.Text
+ black = QtCore.Qt.GlobalColor.black
+ if self.ui.lne_isp.palette().color(text_role) == black:
+ unknown["isp"] = self.ui.lne_isp.text()
+ if self.ui.lne_manufacturer.palette().color(text_role) == black:
+ unknown["manufacturer"] = self.ui.lne_manufacturer.text()
+ return unknown
diff --git a/wg_extensions.py b/wg_extensions.py
new file mode 100644
index 0000000..70e5334
--- /dev/null
+++ b/wg_extensions.py
@@ -0,0 +1,277 @@
+# coding: utf8
+from PySide6 import QtWidgets, QtCore, QtGui
+from util_ext import scan_extensions, ExtensionsData, ProfilesData
+from global_vars import (
+ ExtensionStatusRole,
+ ExtensionIdRole,
+)
+from da_ext_settings import DaExtSettings
+from da_show_profiles import DaShowProfiles
+
+
+class UiWgExtensions(object):
+
+ def __init__(self, window: QtWidgets.QWidget):
+ self.vly_m = QtWidgets.QVBoxLayout()
+ window.setLayout(self.vly_m)
+
+ self.hly_top = QtWidgets.QHBoxLayout()
+ self.vly_m.addLayout(self.hly_top)
+ self.cmbx_browsers = QtWidgets.QComboBox(window)
+ self.cbx_safe = QtWidgets.QCheckBox("安全", window)
+ self.cbx_unsafe = QtWidgets.QCheckBox("不安全", window)
+ self.cbx_unknown = QtWidgets.QCheckBox("未知", window)
+ self.cbx_safe.setChecked(True)
+ self.cbx_unsafe.setChecked(True)
+ self.cbx_unknown.setChecked(True)
+ self.cbx_compat = QtWidgets.QCheckBox("兼容模式", window)
+ self.pbn_update = QtWidgets.QPushButton("更新", window)
+ self.pbn_settings = QtWidgets.QPushButton("设置", window)
+ self.hly_top.addWidget(self.cmbx_browsers)
+ self.hly_top.addWidget(self.cbx_safe)
+ self.hly_top.addWidget(self.cbx_unsafe)
+ self.hly_top.addWidget(self.cbx_unknown)
+ self.hly_top.addStretch(1)
+ self.hly_top.addWidget(self.cbx_compat)
+ self.hly_top.addWidget(self.pbn_update)
+ self.hly_top.addWidget(self.pbn_settings)
+
+ self.lv_extensions = QtWidgets.QListView(window)
+ self.vly_m.addWidget(self.lv_extensions)
+
+
+class BaseExtensionsListModel(QtCore.QAbstractListModel):
+
+ def __init__(self, parent=None):
+ super().__init__(parent)
+ self.all_profiles = {} # type: ProfilesData
+ self.all_extensions = {} # type: ExtensionsData
+ self.names = [] # type: list[tuple[str, str]]
+ self.icons = {} # type: dict[str, QtGui.QIcon]
+ self.safe_info = {} # type: dict[str, dict]
+ self.blank_icon = QtGui.QIcon(":/images/blank_128.png")
+
+ self.last_is_compat = False
+
+ def update(self, is_compat=False):
+ raise NotImplementedError
+
+ def update_ext(self, browser: str, is_compat=False):
+ """内部用"""
+ self.all_profiles.clear()
+ self.all_extensions.clear()
+ self.names.clear()
+ self.icons.clear()
+
+ self.all_extensions, self.all_profiles = scan_extensions(browser, is_compat)
+ self.last_is_compat = is_compat
+ for ext_id in self.all_extensions:
+ name = self.all_extensions[ext_id].name
+ icon = self.all_extensions[ext_id].icon
+ self.names.append((ext_id, name))
+ if len(icon) == 0:
+ self.icons[ext_id] = self.blank_icon
+ else:
+ self.icons[ext_id] = QtGui.QIcon(icon)
+ self.names.sort(key=lambda x: x[1].lower())
+
+ # print("updated " + browser)
+
+ def update_safe_info(self, safe_info: dict):
+ self.safe_info.clear()
+ self.safe_info = safe_info
+
+ def rowCount(self, parent: QtCore.QModelIndex = ...):
+ return len(self.names)
+
+ def data(self, index: QtCore.QModelIndex, role: int = ...):
+ row = index.row()
+ ext_id, name = self.names[row]
+
+ if role == QtCore.Qt.ItemDataRole.DisplayRole:
+ return name
+ if role == QtCore.Qt.ItemDataRole.DecorationRole:
+ return self.icons[ext_id]
+ if role == QtCore.Qt.ItemDataRole.BackgroundRole:
+ is_safe = self.data(index, ExtensionStatusRole)
+ if is_safe is True:
+ return QtGui.QBrush(QtGui.QColor("lightgreen"))
+ elif is_safe is False:
+ return QtGui.QBrush(QtGui.QColor("lightpink"))
+ else:
+ return QtGui.QBrush(QtCore.Qt.BrushStyle.NoBrush)
+ if role == ExtensionStatusRole:
+ if ext_id not in self.safe_info:
+ return None
+ else:
+ return self.safe_info[ext_id]["safe"]
+ if role == ExtensionIdRole:
+ return ext_id
+
+
+class ChromeExtensionsListModel(BaseExtensionsListModel):
+
+ def update(self, is_compat=False):
+ super().update_ext("Chrome", is_compat)
+
+
+class EdgeExtensionsListModel(BaseExtensionsListModel):
+
+ def update(self, is_compat=False):
+ super().update_ext("Edge", is_compat)
+
+
+class BraveExtensionsListModel(BaseExtensionsListModel):
+
+ def update(self, is_compat=False):
+ super().update_ext("Brave", is_compat)
+
+
+class BrowsersListModel(QtCore.QAbstractListModel):
+
+ def __init__(self, parent=None):
+ super().__init__(parent)
+ self.browsers = ["Chrome", "Edge", "Brave"]
+ self.icons = [
+ QtGui.QIcon(":/images/browsers/chrome_32.png"),
+ QtGui.QIcon(":/images/browsers/edge_32.png"),
+ QtGui.QIcon(":/images/browsers/brave_32.png"),
+ ]
+
+ def rowCount(self, parent: QtCore.QModelIndex = ...):
+ return len(self.browsers)
+
+ def data(self, index: QtCore.QModelIndex, role: int = ...):
+ row = index.row()
+
+ if role == QtCore.Qt.ItemDataRole.DisplayRole:
+ return self.browsers[row]
+ if role == QtCore.Qt.ItemDataRole.DecorationRole:
+ return self.icons[row]
+
+
+class WgExtensions(QtWidgets.QWidget):
+
+ def __init__(self, parent=None):
+ super().__init__(parent)
+ self.ui = UiWgExtensions(self)
+
+ self.browsers_list_model = BrowsersListModel(self)
+ self.ui.cmbx_browsers.setModel(self.browsers_list_model)
+
+ self.ext_list_models: dict[str, BaseExtensionsListModel] = {
+ "Chrome": ChromeExtensionsListModel(self),
+ "Edge": EdgeExtensionsListModel(self),
+ "Brave": BraveExtensionsListModel(self),
+ }
+ self.model_is_initial = {
+ "Chrome": True,
+ "Edge": True,
+ "Brave": True,
+ }
+ self.switch_model(self.get_current_browser())
+
+ self.ui.cbx_compat.clicked.connect(self.on_cbx_compat_clicked)
+ self.ui.cmbx_browsers.currentTextChanged.connect(self.on_cmbx_browsers_current_text_changed)
+ self.ui.cbx_safe.clicked.connect(self.on_cbx_safe_clicked)
+ self.ui.cbx_unsafe.clicked.connect(self.on_cbx_unsafe_clicked)
+ self.ui.cbx_unknown.clicked.connect(self.on_cbx_unknown_clicked)
+ self.ui.pbn_update.clicked.connect(self.on_pbn_update_clicked)
+ self.ui.pbn_settings.clicked.connect(self.on_pbn_settings_clicked)
+ self.ui.lv_extensions.doubleClicked.connect(self.on_lv_extensions_double_clicked)
+
+ def on_pbn_settings_clicked(self):
+ da_es = DaExtSettings(self)
+ da_es.exec()
+
+ def get_current_browser(self) -> str:
+ return self.ui.cmbx_browsers.currentData(QtCore.Qt.ItemDataRole.DisplayRole)
+
+ def show_all_rows(self):
+ # 在 update 之前调用
+ self.filters_clicked(True, True)
+ self.filters_clicked(False, True)
+ self.filters_clicked(None, True)
+
+ def apply_rows_hidden(self):
+ # 在 update 之后调用
+ self.filters_clicked(True, self.ui.cbx_safe.isChecked())
+ self.filters_clicked(False, self.ui.cbx_unsafe.isChecked())
+ self.filters_clicked(None, self.ui.cbx_unknown.isChecked())
+
+ def update_model(self, browser: str):
+ model = self.ext_list_models[browser]
+ self.show_all_rows()
+ model.update(is_compat=self.ui.cbx_compat.isChecked())
+ self.apply_rows_hidden()
+
+ def switch_model(self, browser: str):
+ if self.model_is_initial[browser] is True:
+ self.update_model(browser)
+ self.model_is_initial[browser] = False
+
+ model = self.ext_list_models[browser]
+ self.show_all_rows()
+ self.ui.lv_extensions.setModel(model)
+ self.apply_rows_hidden()
+ # 单纯的切换浏览器不一定会导致更新数据,所以需要同步兼容模式的设置
+ self.ui.cbx_compat.setChecked(model.last_is_compat)
+
+ def on_cbx_compat_clicked(self):
+ self.update_model(self.get_current_browser())
+
+ def on_cmbx_browsers_current_text_changed(self, text: str):
+ self.switch_model(text)
+
+ def filters_clicked(self, safe_mark: bool | None, checked: bool):
+ model = self.ext_list_models[self.get_current_browser()]
+ for row in range(model.rowCount()):
+ idx = model.index(row)
+ is_safe = model.data(idx, ExtensionStatusRole)
+ if is_safe is safe_mark:
+ self.ui.lv_extensions.setRowHidden(row, not checked)
+
+ def on_cbx_safe_clicked(self, checked: bool):
+ self.filters_clicked(True, checked)
+
+ def on_cbx_unsafe_clicked(self, checked: bool):
+ self.filters_clicked(False, checked)
+
+ def on_cbx_unknown_clicked(self, checked: bool):
+ self.filters_clicked(None, checked)
+
+ def on_pbn_update_clicked(self):
+ self.update_model(self.get_current_browser())
+
+ def on_lv_extensions_double_clicked(self, index: QtCore.QModelIndex):
+ model = self.ext_list_models[self.get_current_browser()]
+ ext_id = model.data(index, ExtensionIdRole)
+ node = model.all_extensions[ext_id]
+ da_sp = DaShowProfiles(
+ self.get_current_browser(),
+ self.ui.cbx_compat.isChecked(),
+ model.all_profiles,
+ ext_id,
+ node.name,
+ model.icons[ext_id],
+ node.profiles,
+ self
+ )
+ da_sp.exec()
+
+ def update_safe(self, safe_info: dict):
+ for browser in self.ext_list_models:
+ self.ext_list_models[browser].update_safe_info(safe_info)
+
+ def export_unknown(self) -> dict[str, dict]:
+ unknown_ext = {}
+ for browser in self.ext_list_models:
+ model = self.ext_list_models[browser]
+ for row in range(model.rowCount()):
+ idx = model.index(row)
+ is_safe = model.data(idx, ExtensionStatusRole)
+ if is_safe is None:
+ ext_id = model.data(idx, ExtensionIdRole)
+ name = model.data(idx, QtCore.Qt.ItemDataRole.DisplayRole)
+ unknown_ext[ext_id] = {"name": name}
+ return unknown_ext
diff --git a/wg_software.py b/wg_software.py
new file mode 100644
index 0000000..a4ed62e
--- /dev/null
+++ b/wg_software.py
@@ -0,0 +1,131 @@
+# coding: utf8
+from PySide6 import QtWidgets, QtCore, QtGui
+from util_func import (
+ get_win_installed_software,
+ extract_win_icon_from_file,
+)
+from global_vars import (
+ SoftwareStatusRole,
+)
+
+
+class UiWgSoftware(object):
+
+ def __init__(self, window: QtWidgets.QWidget):
+ self.vly_m = QtWidgets.QVBoxLayout()
+ window.setLayout(self.vly_m)
+
+ self.hly_top = QtWidgets.QHBoxLayout()
+ self.vly_m.addLayout(self.hly_top)
+ self.cbx_safe = QtWidgets.QCheckBox("安全", window)
+ self.cbx_unsafe = QtWidgets.QCheckBox("不安全", window)
+ self.cbx_unknown = QtWidgets.QCheckBox("未知", window)
+ self.cbx_safe.setChecked(True)
+ self.cbx_unsafe.setChecked(True)
+ self.cbx_unknown.setChecked(True)
+ self.hly_top.addWidget(self.cbx_safe)
+ self.hly_top.addWidget(self.cbx_unsafe)
+ self.hly_top.addWidget(self.cbx_unknown)
+ self.hly_top.addStretch(1)
+
+ self.lv_software = QtWidgets.QListView(window)
+ self.vly_m.addWidget(self.lv_software)
+
+
+class SoftwareListModel(QtCore.QAbstractListModel):
+
+ def __init__(self, parent=None):
+ super().__init__(parent)
+ self.all_software = {} # type: dict[str, str]
+ self.names = [] # type: list[str]
+ self.icons = {} # type: dict[str, QtGui.QIcon]
+ self.safe_info = {}
+ self.blank_icon = QtGui.QIcon(":/images/blank_128.png")
+ self.update()
+
+ def update(self):
+ self.all_software.clear()
+ self.names.clear()
+ self.icons.clear()
+
+ self.all_software = get_win_installed_software()
+ for s in self.all_software:
+ self.names.append(s)
+ self.icons[s] = extract_win_icon_from_file(
+ self.all_software[s],
+ self.blank_icon
+ )
+ self.names.sort(key=lambda x: x.lower())
+
+ def update_safe_info(self, safe_info: dict):
+ self.safe_info.clear()
+ self.safe_info = safe_info
+
+ def rowCount(self, parent: QtCore.QModelIndex = ...):
+ return len(self.names)
+
+ def data(self, index: QtCore.QModelIndex, role: int = ...):
+ row = index.row()
+ name = self.names[row]
+
+ if role == QtCore.Qt.ItemDataRole.DisplayRole:
+ return name
+ if role == QtCore.Qt.ItemDataRole.DecorationRole:
+ return self.icons[name]
+ if role == QtCore.Qt.ItemDataRole.BackgroundRole:
+ is_safe = self.data(index, SoftwareStatusRole)
+ if is_safe is True:
+ return QtGui.QBrush(QtGui.QColor("lightgreen"))
+ elif is_safe is False:
+ return QtGui.QBrush(QtGui.QColor("lightpink"))
+ else:
+ return QtGui.QBrush(QtCore.Qt.BrushStyle.NoBrush)
+ if role == SoftwareStatusRole:
+ if name not in self.safe_info:
+ return None
+ else:
+ return self.safe_info[name]["safe"]
+
+
+class WgSoftware(QtWidgets.QWidget):
+
+ def __init__(self, parent=None):
+ super().__init__(parent)
+ self.ui = UiWgSoftware(self)
+
+ self.software_list_model = SoftwareListModel(self)
+ self.ui.lv_software.setModel(self.software_list_model)
+
+ self.ui.cbx_safe.clicked.connect(self.on_cbx_safe_clicked)
+ self.ui.cbx_unsafe.clicked.connect(self.on_cbx_unsafe_clicked)
+ self.ui.cbx_unknown.clicked.connect(self.on_cbx_unknown_clicked)
+
+ def filters_clicked(self, safe_mark: bool | None, checked: bool):
+ for row in range(self.software_list_model.rowCount()):
+ idx = self.software_list_model.index(row)
+ is_safe = self.software_list_model.data(idx, SoftwareStatusRole)
+ if is_safe is safe_mark:
+ self.ui.lv_software.setRowHidden(row, not checked)
+
+ def on_cbx_safe_clicked(self, checked: bool):
+ self.filters_clicked(True, checked)
+
+ def on_cbx_unsafe_clicked(self, checked: bool):
+ self.filters_clicked(False, checked)
+
+ def on_cbx_unknown_clicked(self, checked: bool):
+ self.filters_clicked(None, checked)
+
+ def update_safe(self, safe_info: dict):
+ self.software_list_model.update_safe_info(safe_info)
+
+ def export_unknown(self) -> list[str]:
+ unknown_software = []
+ for row in range(self.software_list_model.rowCount()):
+ idx = self.software_list_model.index(row)
+ is_safe = self.software_list_model.data(idx, SoftwareStatusRole)
+ if is_safe is None:
+ name = self.software_list_model.data(idx, QtCore.Qt.ItemDataRole.DisplayRole)
+ unknown_software.append(name)
+
+ return unknown_software