[
  {
    "path": ".github/workflows/main.yml",
    "content": "name: Release CI\non: push\njobs:\n  x64_build:\n    runs-on: windows-2019\n\n    steps:\n    - uses: actions/checkout@v2\n      \n    - name: Add msbuild to PATH\n      uses: microsoft/setup-msbuild@v1\n      \n    - name: Run msbuild\n      run: msbuild -p:configuration=release -p:platform=x64 -p:platformToolset=v142\n      \n    - name: Get current time\n      uses: 1466587594/current-time@v1\n      id: current-time\n      with:\n        format: YYYYMMDD_HHmmss\n        utcOffset: \"+08:00\"\n      \n    - name : Upload artifact\n      uses: actions/upload-artifact@v2\n      with:\n        name: x64_${{ steps.current-time.outputs.formattedTime }}_TrafficMonitor\n        path: |\n          Bin/x64/Release/TrafficMonitor.exe\n          Bin/x64/Release/*.dll\n\n    - name : Upload pdb files\n      uses: actions/upload-artifact@v2\n      with:\n        name: x64_${{ steps.current-time.outputs.formattedTime }}_pdb\n        path: Bin/x64/Release/*.pdb\n\n  x86_build:\n    runs-on: windows-2019\n\n    steps:\n    - uses: actions/checkout@v2\n      \n    - name: Add msbuild to PATH\n      uses: microsoft/setup-msbuild@v1\n      \n    - name: Run msbuild\n      run: msbuild -p:configuration=release -p:platform=x86 -p:platformToolset=v142\n      \n    - name: Get current time\n      uses: 1466587594/current-time@v1\n      id: current-time\n      with:\n        format: YYYYMMDD_HHmmss\n        utcOffset: \"+08:00\"\n      \n    - name : Upload artifact\n      uses: actions/upload-artifact@v2\n      with:\n        name: x86_${{ steps.current-time.outputs.formattedTime }}_TrafficMonitor\n        path: |\n          Bin/Release/TrafficMonitor.exe\n          Bin/Release/*.dll\n      \n    - name : Upload pdb files\n      uses: actions/upload-artifact@v2\n      with:\n        name: x86_${{ steps.current-time.outputs.formattedTime }}_pdb\n        path: Bin/Release/*.pdb\n\n  # winXP_build:\n    # runs-on: windows-latest\n\n    # steps:\n    # - uses: actions/checkout@v2\n      \n    # - name: Add msbuild to PATH\n      # uses: microsoft/setup-msbuild@v1\n      \n    # - name: Run msbuild\n      # run: |\n        # set ExternalCompilerOptions=/DCOMPILE_FOR_WINXP\n        # msbuild -p:configuration=release -p:platform=x86 -p:platformToolset=v140_xp\n      # shell: cmd\n      \n    # - name: Get current time\n      # uses: 1466587594/current-time@v1\n      # id: current-time\n      # with:\n        # format: YYYYMMDD_HHmmss\n        # utcOffset: \"+08:00\"\n      \n    # - name : Upload artifact\n      # uses: actions/upload-artifact@v1\n      # with:\n        # name: winXP_${{ steps.current-time.outputs.formattedTime }}_TrafficMonitor\n        # path: Bin/Release/TrafficMonitor.exe\n"
  },
  {
    "path": ".gitignore",
    "content": "## Ignore Visual Studio temporary files, build results, and\n## files generated by popular Visual Studio add-ons.\n##\n## Get latest from https://github.com/github/gitignore/blob/master/VisualStudio.gitignore\n\n# User-specific files\n*.suo\n*.user\n*.userosscache\n*.sln.docstates\n\n# User-specific files (MonoDevelop/Xamarin Studio)\n*.userprefs\n\n# Build results\n[Dd]ebug/\n[Dd]ebugPublic/\n[Rr]elease/\n[Rr]eleases/\nx64/\nx86/\nbld/\n[Bb]in/\n[Oo]bj/\n[Ll]og/\n\n# Visual Studio 2015 cache/options directory\n.vs/\n# Uncomment if you have tasks that create the project's static files in wwwroot\n#wwwroot/\n\n# MSTest test Results\n[Tt]est[Rr]esult*/\n[Bb]uild[Ll]og.*\n\n# NUNIT\n*.VisualState.xml\nTestResult.xml\n\n# Build Results of an ATL Project\n[Dd]ebugPS/\n[Rr]eleasePS/\ndlldata.c\n\n# .NET Core\nproject.lock.json\nproject.fragment.lock.json\nartifacts/\n**/Properties/launchSettings.json\n\n*_i.c\n*_p.c\n*_i.h\n*.ilk\n*.meta\n*.obj\n*.pch\n*.pdb\n*.pgc\n*.pgd\n*.rsp\n*.sbr\n*.tlb\n*.tli\n*.tlh\n*.tmp\n*.tmp_proj\n*.log\n*.vspscc\n*.vssscc\n.builds\n*.pidb\n*.svclog\n*.scc\n\n# Chutzpah Test files\n_Chutzpah*\n\n# Visual C++ cache files\nipch/\n*.aps\n*.ncb\n*.opendb\n*.opensdf\n*.sdf\n*.cachefile\n*.VC.db\n*.VC.VC.opendb\n\n# Visual Studio profiler\n*.psess\n*.vsp\n*.vspx\n*.sap\n\n# TFS 2012 Local Workspace\n$tf/\n\n# Guidance Automation Toolkit\n*.gpState\n\n# ReSharper is a .NET coding add-in\n_ReSharper*/\n*.[Rr]e[Ss]harper\n*.DotSettings.user\n\n# JustCode is a .NET coding add-in\n.JustCode\n\n# TeamCity is a build add-in\n_TeamCity*\n\n# DotCover is a Code Coverage Tool\n*.dotCover\n\n# Visual Studio code coverage results\n*.coverage\n*.coveragexml\n\n# NCrunch\n_NCrunch_*\n.*crunch*.local.xml\nnCrunchTemp_*\n\n# MightyMoose\n*.mm.*\nAutoTest.Net/\n\n# Web workbench (sass)\n.sass-cache/\n\n# Installshield output folder\n[Ee]xpress/\n\n# DocProject is a documentation generator add-in\nDocProject/buildhelp/\nDocProject/Help/*.HxT\nDocProject/Help/*.HxC\nDocProject/Help/*.hhc\nDocProject/Help/*.hhk\nDocProject/Help/*.hhp\nDocProject/Help/Html2\nDocProject/Help/html\n\n# Click-Once directory\npublish/\n\n# Publish Web Output\n*.[Pp]ublish.xml\n*.azurePubxml\n# TODO: Comment the next line if you want to checkin your web deploy settings\n# but database connection strings (with potential passwords) will be unencrypted\n*.pubxml\n*.publishproj\n\n# Microsoft Azure Web App publish settings. Comment the next line if you want to\n# checkin your Azure Web App publish settings, but sensitive information contained\n# in these scripts will be unencrypted\nPublishScripts/\n\n# NuGet Packages\n*.nupkg\n# The packages folder can be ignored because of Package Restore\n**/packages/*\n# except build/, which is used as an MSBuild target.\n!**/packages/build/\n# Uncomment if necessary however generally it will be regenerated when needed\n#!**/packages/repositories.config\n# NuGet v3's project.json files produces more ignorable files\n*.nuget.props\n*.nuget.targets\n\n# Microsoft Azure Build Output\ncsx/\n*.build.csdef\n\n# Microsoft Azure Emulator\necf/\nrcf/\n\n# Windows Store app package directories and files\nAppPackages/\nBundleArtifacts/\nPackage.StoreAssociation.xml\n_pkginfo.txt\n\n# Visual Studio cache files\n# files ending in .cache can be ignored\n*.[Cc]ache\n# but keep track of directories ending in .cache\n!*.[Cc]ache/\n\n# Others\nClientBin/\n~$*\n*~\n*.dbmdl\n*.dbproj.schemaview\n*.jfm\n*.pfx\n*.publishsettings\norleans.codegen.cs\n\n# Since there are multiple workflows, uncomment next line to ignore bower_components\n# (https://github.com/github/gitignore/pull/1529#issuecomment-104372622)\n#bower_components/\n\n# RIA/Silverlight projects\nGenerated_Code/\n\n# Backup & report files from converting an old project file\n# to a newer Visual Studio version. Backup files are not needed,\n# because we have git ;-)\n_UpgradeReport_Files/\nBackup*/\nUpgradeLog*.XML\nUpgradeLog*.htm\n\n# SQL Server files\n*.mdf\n*.ldf\n*.ndf\n\n# Business Intelligence projects\n*.rdl.data\n*.bim.layout\n*.bim_*.settings\n\n# Microsoft Fakes\nFakesAssemblies/\n\n# GhostDoc plugin setting file\n*.GhostDoc.xml\n\n# Node.js Tools for Visual Studio\n.ntvs_analysis.dat\nnode_modules/\n\n# Typescript v1 declaration files\ntypings/\n\n# Visual Studio 6 build log\n*.plg\n\n# Visual Studio 6 workspace options file\n*.opt\n\n# Visual Studio 6 auto-generated workspace file (contains which files were open etc.)\n*.vbw\n\n# Visual Studio LightSwitch build output\n**/*.HTMLClient/GeneratedArtifacts\n**/*.DesktopClient/GeneratedArtifacts\n**/*.DesktopClient/ModelManifest.xml\n**/*.Server/GeneratedArtifacts\n**/*.Server/ModelManifest.xml\n_Pvt_Extensions\n\n# Paket dependency manager\n.paket/paket.exe\npaket-files/\n\n# FAKE - F# Make\n.fake/\n\n# JetBrains Rider\n.idea/\n*.sln.iml\n\n# CodeRush\n.cr/\n\n# Python Tools for Visual Studio (PTVS)\n__pycache__/\n*.pyc\n\n# Cake - Uncomment if you are using it\n# tools/**\n# !tools/packages.config\n\n# Telerik's JustMock configuration file\n*.jmconfig\n\n# BizTalk build output\n*.btp.cs\n*.btm.cs\n*.odx.cs\n*.xsd.cs\n\n*.ini\n!skin.ini\n*.dat\n*.bak\n/TrafficMonitor/skins/test*\n[Dd]ebug (without temperature)/\n[Rr]elease (without temperature)/\n[Dd]ebug (lite)/\n[Rr]elease (lite)/\n/TrafficMonitor/compile_time.txt\n"
  },
  {
    "path": "Help.md",
    "content": "**简体中文 | [English](./Help_en-us.md)**<br>\n\n# TrafficMonitor 常见问题\n这里是关于TrafficMonitor常见问题的页面，如果你想查看关于TrafficMonitor的各项功能和使用方法的详细介绍，请[点击这里](https://github.com/zhongyang219/TrafficMonitor/wiki)转到TraffinMonitor Wiki页面。\n\n### 1. 如何显示CPU和内存利用率？\n\n在主窗口点击右弹出键菜单，勾选“显示更多信息”。如果需要在任务栏窗口中也显示CPU和内存利用率，则在任务栏窗口中点击右键弹出菜单，在“显示设置<img src=\"./Screenshots/images/item.png\" style=\"zoom: 80%;\" />”子菜单下勾选“CPU和内存利用率”即可。\n### 2. 如何单独设置任务栏窗口中每个项目的颜色？\n在右键菜单中选择“选项”，切换到“任务栏窗口设置”，勾选“指定每个项目的颜色”，此时再点击“文本颜色”右边的颜色块，就会弹出“任务栏窗口颜色设置”的对话框了。<br>\n如果不勾选“指定每个项目的颜色”，则只能为文本设置统一的颜色。\n\n### 3. 设置了开机自动运行仍然无法开机自启。\n从1.80版本开始，包含温度监控的版本和不含温度监控的版本采用了不同的方式来实现开机自启动。\n\n* 不含温度监控的版本和1.80以前的版本：\n\n  不含温度监控的版本和1.80以前的版本的开机启动功能是通过在注册表“计算机\\HKEY_CURRENT_USER\\Software\\Microsoft\\Windows\\CurrentVersion\\Run”中创建“TrafficMonitor”的注册表项来实现的，如果你遇到无法开机启动的问题，请先检查该注册表项是否存在，再检查程序的路径是否正确。如果你移动了程序的位置，则会因为路径无效导致开机自启动失效，此时只需要在选项设置中取消“开机自动运行”的勾选，再将其勾选上就可以了。<br>\n\n  注意，某些第三方安全软件可能会阻止TrafficMonitor的开机自启动，请尝试在安全软件中允许TrafficMonitor开机自启动。\n\n  如果设置了以管理员身份运行也会出现开机无法自启动的问题，请尝试去掉以管理员身份运行。<br>\n\n* 包含温度监控的版本：\n\n  包含温度监控的版本是通过创建任务计划来实现开机自启动的。可以通过`控制面板\\系统和安全\\管理工具`来打开任务计划程序。\n\n  如下图所示：\n\n  <img src=\"./Screenshots/images/image3.jpg\"/>\n\n  如果你遇到无法开机自启动的情况，请到“任务计划程序”中检查TrafficMonitor的计划任务是否正常创建，exe文件的路径是否正确。\n\n开机无法自启动的一个常见原因是你可能移动了TrafficMonitor主程序的位置。如果你为TrafficMonitor设置好了开机自动运行，但是你将TrafficMonitor移动到了其他位置，那么开机自启动肯定就失效了。你需要打开TrafficMonitor的“选项设置”——“常规设置”，如果“开机时自动运行”处于勾选状态，先去掉勾选，然后再次打开“选项设置”——“常规设置”，重新勾选“开机时自动运行”即可。\n\n需要注意的是，如果你使用不含温度监控的版本在注册表中创建了开机自启动项，然后再使用包含温度监控的版本开启开机自启动功能，它会自动将注册表中的开机自启动项删除，再在任务计划中创建开机自启动项。反之亦然。\n\n### 4. 程序弹出“无法保存设置”的对话框。\n如果遇到这种情况，说明程序没有向其所在目录下写入数据的权限，导致设置数据无法保存。尝试将程序移动到其他有写入权限的文件夹中可以解决这个问题。<br>\n你也可以通过以下步骤将配置文件的保存路径改为C:\\Users\\\\<用户名\\>\\AppData\\Roaming\\TrafficMonitor目录。<br>\n\n* 退出TrafficMonitor，以管理员身份重新启动TrafficMonitor。\n* 在右键菜单中选择“选项”，切换到“常规设置”选项卡，在“数据和配置文件”中选择“保存到Appdata目录”。<br>\n\n如果此时仍然提示“无法保存设置”，请打开应用程序所在目录，打开`global_cfg.ini`文件，如果不存在请新建一个，在里面添加如下内容：\n\n```\n[config]\nportable_mode = true\n```\n\n如果无法新建，可以在其他位置（比如桌面）新建该文件，然后移动到程序所在目录。\n\n如果`global_cfg.ini`文件已存在，就把`portable_mode `的值改成`true`，保存后重新启动TrafficMonitor。\n\n如果`global_cfg.ini`没有写入权限，可以尝试把该文件复制到桌面，改好后复制回原来的路径将原文件覆盖。\n\n执行以上步骤后理论上应该不会出现这种问题了。如果这个问题仍然出现，请尝试把C:\\Users\\\\<用户名\\>\\AppData\\Roaming\\TrafficMonitor\\config.ini删除，该文件会在删除后重新生成。\n\n在1.79以后的版本中，如果程序所在目录无法写入数据，会自动将配置和数据文件保存到C:\\Users\\\\<用户名\\>\\AppData\\Roaming\\TrafficMonitor目录，此时，“选项”——“常规设置”——“数据和配置文件”中“保存到程序所在目录”将不可用。\n\n### 5. 多显示器时悬浮窗只能在主显示器中显示。\n默认情况下做了让悬浮窗无法超出屏幕边界的处理，如果需要将悬浮窗移动到其他显示器，请在悬浮窗中点击右键弹出菜单，选择“其他功能”，勾选“允许超出屏幕边界”，此时悬浮窗不再限制在屏幕内，也能移动到任意显示器中了。<br>\n如果移除额外的显示器，那么悬浮窗可能会出现在屏幕区域外导致不可见，此时只需要在通知区域图标上点击鼠标右键，选择“其他功能”，去掉“允许超出屏幕边界”的勾选，悬浮窗就会重新出现屏幕范围内了。\n\n### 6. 网速数值显示不全。\n由于不同字体每个字符的宽度并不一样，在某些情况下，确实会出现网速数值显示不全的问题。如果出现了这样的问题，请打开“选项”——“任务栏窗口设置”，在“数据位数”下拉列表中选择一个更大的值。\n### 7. 设置了鼠标穿透后如何取消？\n在通知区的TrafficMonitor的图标上点击鼠标右键，去掉“鼠标穿透”的勾选即可。<br>\n设置了鼠标穿透后，悬浮窗将无法响应任何鼠标消息，也无法弹出右键菜单，但是可以通过通知区图标来弹出右键菜单。主窗口的右键菜单和通知区图标的右键是完全一样的。<br>\n另外，即使你之前设置了隐藏通知区图标，开启鼠标穿透后，通知区图标也会自动显示出来，防止无法弹出右键菜单。<br>\n说明：以下几种情况下通知区图标会自动显示出来：<br>\n\n* 开启鼠标穿透后；\n* 不显示任务栏窗口的情况下隐藏主窗口后；\n* 隐藏主窗口的情况下关闭任务栏窗口后；\n* 开启鼠标穿透的情况下关闭任务栏窗口后。\n### 8. Windows 10 白色任务栏主题时任务栏窗口颜色的问题\n\n在使用白色任务栏主题时，你可以在在“任务栏窗口设置”点击“预设方案”按钮，选择“浅色模式”，可以一键设置浅色模式任务栏颜色。如图所示：\n\n<img src=\"./Screenshots/images/image2.jpg\" style=\"zoom:80%;\" />\n\n同时，你还可能勾选“自动适应Windows10深色/浅色主题”，程序会在Windows10深色/浅色主题更换时自动切换颜色方案。你可以点击“自动适应设置”按钮来配置在深色和浅色主题时分别使用哪个颜色预设。\n\n### 9. 在Windows7/Windows8/8.1下任务栏窗口有个背景色，无法完全透明\n这个问题确实存在，但是在Win10下是正常的。这个问题暂时无法解决。\n\n在1.79以后的版本中，Windows8/8.1下可以在“选项”——“任务栏窗口设置”中勾选“背景色透明”，再勾选“根据任务栏颜色自动设置背景色”即可获得较好的显示效果。\n\n### 10. 任务栏窗口有时会显示不全，比如单位被覆盖了\n这确实是一个BUG，但是我目前还没有找到一个好的解决方法，这个问题通常出现在任务栏右侧通知区域宽度变化的时候，主要在在切换输入法的时候，如果出现了这个问题，可以将通知区任意一个图标向上拖动将其隐藏，再将其拖下来即可恢复正常。<br>\n出现这个问题原因在于，由于系统任务栏通知区的图标数量可能会发生变化，导致通知区的宽度也会时常变化，当通知区的宽度发生变化时，TrafficMonitor的任务栏窗口需要实时调整其位置。但是由于我无法知道通知区的宽度在什么时候变化，因此只能每隔一段时间判断是否需要调整位置，如果任务栏通知区域的宽度变化得太快，就会导致TrafficMonitor的任务栏无法及时调整其位置，从而导致了这个BUG。<br>\n\n#### 以下步骤或许可以解决这个问题：<br>\n* 打开“设置”\n* 点击“时间和语言”——“区域和语言”\n* 点击右侧“高级键盘设置”\n* 勾选“使用桌面语言栏”\n* 右键点击任务栏，选择“任务栏设置”\n* 点击“打开或关闭系统图标”，关闭“输入指示”<br>\n方法来自知乎 [win10的任务栏为何一点击就乱动？](https://www.zhihu.com/question/312032145/answer/627965084)<br>\n\n### 11. Windows10中开启HDR后任务栏窗口无法显示\n\n部分用户反馈，在Windows10中开始HDR功能会导致任务栏窗口无法显示。如果遇到这个问题，可以尝试在[“选项设置”——“任务栏窗口设置”](https://github.com/zhongyang219/TrafficMonitor/wiki/选项设置#任务栏窗口设置)中关闭“背景透明”选项的勾选。\n\n### 12. CPU利用率显示和任务管理器不一致\n\n在Windows10及以上操作系统中，如果你需要让TrafficMonitor显示的CPU利用率和任务管理器一致，请到[“选项设置”——“常规设置”——“高级”](https://github.com/zhongyang219/TrafficMonitor/wiki/选项设置#高级)——“CPU使用率获取方式”中选择“使用性能计数器”。\n\n由于Windows10以上操作系统中任务管理器获取CPU利用率的方式发生了改变，因此选择“基于CPU使用时间”的方式获取到的CPU利用率会和任务管理器中显示的不一致。\n\n### 13. 关于TrafficMonitor温度监控的问题\n\n由于温度监控功能在某些电脑中存在一些问题，因此温度监控功能默认是关闭的，如果你要使用TrafficMonitor的温度监控功能，请到[“选项设置”-“常规设置”-“硬件监控”](https://github.com/zhongyang219/TrafficMonitor/wiki/选项设置#硬件监控)中开启。开启后，任务栏右键菜单中的“显示设置”子菜单下才会出现温度相关的项目。\n\nTrafficMonitor的温度监控功能依赖第三方开源库[LibreHardwareMonitor](https://github.com/LibreHardwareMonitor/LibreHardwareMonitor)。如果你遇到硬件温度无法显示，或者显示的温度异常的问题，请先下载LibreHardwareMonitor，并查看LibreHardwareMonitor是否能正常显示对应的温度。\n\n如果LibreHardwareMonitor也无法显示对应硬件的温度，那么我也无法解决这个问题，你可以向LibreHardwareMonitor作者反馈你的问题。\n\n如果LibreHardwareMonitor可以正常显示对应硬件的温度，请向我反馈这个问题，同时提供LibreHardwareMonitor的截图，以便我调查你的问题。\n\n**注意：硬件监控功能（包括温度监控和显卡使用率监控）可能存在一些问题，它可能会占用更多的CPU和内存。据部分用户反馈，开启温度功能后会导致程序崩溃和系统死机等问题，请在知晓以上风险后再决定开启硬件监控功能。否则，请不要使用硬件监控功能。**\n\n## 14. 程序启动时提示找不到“MSVC\\*.dll”或“mfc\\*.dll”\n\n点击以下链接下载并安装Microsoft Visual C++ 运行环境。\n\n[最新支持的 Visual C++ 可再发行程序包下载 | Microsoft Docs](https://docs.microsoft.com/zh-CN/cpp/windows/latest-supported-vc-redist?view=msvc-170)\n\n<br>\n\n>如果还遇到其他问题，请点击“关于”对话框中的“联系作者”，或者直接[点击此处](mailto:zhongyang219@hotmail.com)向我发送电子邮件。但我由于作者的能力有限，我并不能保证可以解决所有问题，但是你的反馈也许可以帮助我更好的改进这个软件。<br>\n请在邮件中尽可能详细地描述你遇到的问题，出现了什么错误提示，你尝试过哪些操作等，最好能够附上截图和配置文件（“选项”——“常规设置”——“打开配置文件所在路径”）。<br>\n注意，发送前请先确认一下你发送时使用的电子邮件地址，如果你的邮件地址是形如“outlook_随机字符串@outlook.com”的格式，那么这样的邮箱地址是无法回复的。\n这可能是由于你使用了第三方邮箱地址作为Microsoft账号登录Windows导致的。如果有这样的情况，请务必在邮件中附上正确的电子邮件地址。\n"
  },
  {
    "path": "Help_en-us.md",
    "content": "**[简体中文](./Help.md) | English**<br>\n# TrafficMonitor Frequently Asked Questions\nThis is the page about the Frequently Asked Questions of TrafficMonitor. If you want to view the detailed introduction of the functions and usage of TrafficMonitor, please [click here](https://github.com/zhongyang219/TrafficMonitor/wiki) to go the the TraffinMonitor Wiki page (Only Chinese Available).\n\n### 1. How to show the CPU and memory usage?\n\nRight click the main window and check \"Show More Info\". If you also need to display the CPU and memory usage in the taskbar window, right click the taskbar window and check \"CPU and Memory Usage\" in the \"Display Settings<img src=\"./Screenshots/images/item.png\" style=\"zoom: 80%;\" />\" sub menu.\n### 2. How do I set the color of each item in the taskbar window individually?\nSelect \"Options\" In the right-click menu, switch to \"Taskbar Window Settings\" tab, check \"Specify colors for each item\", and then click the color block on the right side of \"Text Color \" to pop up the dialog box for taskbar window color settings. <br>\nIf you do not check \"Specify colors for each item\", you can only set the uniform color for the text.\n### 3. \"Auto run when Windows start\" dose not work\nStarting from version 1.80, the version with temperature monitoring and the version without temperature monitoring have adopted different methods to realize \"auto run when Windows start\".\n\n* Versions without temperature monitoring and versions before 1.80:\n\n  The auto run function of the versions without temperature monitoring and versions before 1.80 is achieved by creating the \"TrafficMonitor\" key in the registry path of \"Computer\\HKEY_CURRENT_USER\\Software\\Microsoft\\Windows\\CurrentVersion\\Run\".\n  If you encounter the problem of auto run dose not work, please check that the registry exists, and then check the program path is correct. If you move the location of the program, the auto run will be invalid because of the program location being invalid. In this case, you only need to uncheck \"Auto run when Windows start\" in the option settings, and then check it on.<br>\n\n  Note that some third-party security software may prevent TrafficMonitor from booting up automatically. Please try to allow TrafficMonitor to boot up automatically in the security software.\n\n  If you set the program to run as an administrator, the auto run function will also not work. Please try to remove running as an administrator.<br>\n\n* Version with temperature monitoring:\n\n  The version that includes temperature monitoring implements auto run function by creating a task plan. The task scheduler can be opened through `Control Panel\\System and Security\\Administrative Tools`.  As shown below:\n\n  <img src=\"./Screenshots/images/image3.jpg\"/>\n\n  If you encounter that situation that the program cannot be started automatically after booting, please go to \"Task Scheduler\" to check whether the scheduled task of TrafficMonitor is created normally and the path of the exe file is correct.\n\nA common reason why the \"Auto run when Windows start\" dose not work is that you may have moved the location of the TrafficMonitor main program. If you have set up TrafficMonitor to run automatically at startup, but you move TrafficMonitor to another location, then the auto run at startup will definitely fail. You need to open \"Option Settings\"-\"General Settings\" of TrafficMonitor, if \"Auto run when Windows start\" is checked, remove the check, then open \"Option Settings\"-\"General Settings\" again, and check again \"Auto run when Windows start\" will be fine.\n\nIt should be noted that if you use the version without temperature monitoring to create a auto-start item in the registry, and then use the version that with temperature monitoring to turn on the \"Auto run when Windows start\" function, it will automatically delete the auto-start item in the registry, and then create a auto-start item in the task plan. vice versa.\n\n### 4. The program pops up the \"Unable to Save Settings\" message box.\nIf you encountered this problem, that means the program does not have permission to write data to its directory, causing the settings data cannot be saved. Try to move the program to another folder that has write permissions will save this problem. <br>\nYou can also change the save path of the configuration and data file to the C:\\Users\\\\<username\\>\\AppData\\Roaming\\TrafficMonitor directory by the following steps. <br>\n* Exit TrafficMonitor and restart TrafficMonitor as an administrator.\n* Select \"Options\" in the right-click menu, switch to the \"General Settings\" tab, and select \"Save to Appdata Directory\" in \"Configuration and data files\".\n\nIf it still pops up the \"Unable to Save Settings\" message box, open the directory where the application is located, open the `global_cfg.ini` file. If it doesn't exist, create a new one, adding the following:\n\n```\n[config]\nportable_mode = true\n```\n\nIf the file cannot be created, you can create the file in a different location, such as the desktop, and then move to the directory where the program is located.\n\nIf the `global_cfg.ini` file already exists, change the value of `portable_mode` to `true` and save, then restart TrafficMonitor.\n\nIf `global_cfg.ini` does not have write permission, you can try copying the file to the desktop, and then copying it back to the original path to overwrite the original file after modified.\n\nAfter these steps, this problem should not theoretically occur. If this problem still occurs, try to delete the file C:\\Users\\\\<username\\>\\AppData\\Roaming\\TrafficMonitor\\config.ini. This file will be regenerated after it is deleted. \n\nIn version 1.79 and later, if the directory where the program located is not writable, the configuration and data files will be automatically saved to the \"C:\\Users\\\\<username\\>\\AppData\\Roaming\\TrafficMonitor\" directory. At this time, \"Options\" -“General Settings”-“Save to the program directory” in “Configuration and data files” will not be available.\n\n### 5. The suspension window can only be displayed on the main monitor when multiple monitors are present.\nBy default, the suspension window cannot be moved out of the screen boundaries. If you need to move the suspension window to other monitors, please right click the suspension window, select \"Other Functions\", check \"Allow Out of Screen Boundaries\". At this time the suspension window is no longer limited to the screen, it can be also move to any monitors. <br>\nIf you remove the extra monitor, then the suspension window may appear outside the screen area to cause the invisible. At this time, you only need to right click the the icon in the notification area, select \"Other function\", uncheck \"Allow Out of Screen Boundaries\", the suspension window will appear in the screen area again.\n### 6. The net speed value is not fully displayed.\nBecause the width of each character in different fonts is not the same, in some cases, it does appear the problem of the net speed value is not fully displayed. If this problem occurs, open \"Options\"-\"Taskbar Window Settings\", and select a larger value in the \"Number of digits\" drop down list.\n### 7. How to cancel after setting the mouse penetrate?\nRight click the TrafficMonitor icon in the notification area, uncheck the \"Mouse Penetrate\".<br>\nAfter the mouse penetrate is set, the suspension window will not be able to respond to any mouse messages, or pop-up right-click menu. But the menu can be poped up by right click the icon in notification area. The right-click menu of the main window and which of the notification area icon is exactly the same.<br>\nIn addition, even if you have previously set the hidden notify icon, when the \"mouse penetrate\" is set, the notify icon will also automatically appear to prevent the right-click menu cannot be displayed. <br>\nNote: The notification area icon will be automatically displayed in the following situations: <br>\n* After the \"Mouse Penetrate\" is set;\n* Hide the main window without displaying the taskbar window;\n* Close the taskbar window when the main window is hidden;\n* Close the taskbar window when the \"Mouse Penetrate\" is set.\n### 8. Problems with the taskbar window color in Windows 10 white taskbar theme\nWhen using the white taskbar theme, you can click the \"Preset\" button in the \"Taskbar Window Settings\" and select \"Light Mode\" to set the taskbar color in light mode. As the picture shows:\n\n<img src=\"./Screenshots/images/image2.jpg\" style=\"zoom:80%;\" />\n\nAt the same time, you may also check \"Auto adapt to Windows 10 dark/light theme\", the program will automatically switch the color preset when Windows 10 dark/light theme is changed. You can click the \"Auto Adapt settings\" button to configure which color preset to use for dark and light themes.\n\n### 9. The taskbar windows cannot be displayed when the HDR is turned on in Windows 10\n\nSome users have reported that turning on the HDR function in Windows 10 will cause the taskbar window to fail to display. If you encounter this problem, you can try turning off the \"Background Transparent\" option in [\"Option Settings\" - \"Taskbar Window Settings\"](https://github.com/zhongyang219/TrafficMonitor/wiki/选项设置#任务栏窗口设置).\n\n### 13. About the temperature monitoring of TrafficMonitor\n\nDue to some problems occurred in some computers caused by the temperature monitoring function, the temperature monitoring function is turned off by default. If you want to use the temperature monitoring function of TrafficMonitor, please go to [\"Option Settings\"-\"General Settings\"-\"Hardware Monitoring\"](https://github.com/zhongyang219/TrafficMonitor/wiki/选项设置#硬件监控) to enable it. After it is turned on, temperature-related items will appear under the \"Display Settings\" submenu in the taskbar right-click menu.\n\nThe temperature monitoring function of TrafficMonitor relies on a third-party open source library [LibreHardwareMonitor](https://github.com/LibreHardwareMonitor/LibreHardwareMonitor). If you encounter the problem that the hardware temperature cannot be displayed, or the displayed temperature is incorrect, then  please download LibreHardwareMonitor first, and check whether LibreHardwareMonitor can display the corresponding temperature normally.\n\nIf LibreHardwareMonitor cannot display the temperature of the corresponding hardware, then I cannot solve this problem. You can report your problem to the author of LibreHardwareMonitor.\n\nIf LibreHardwareMonitor can display the temperature of the corresponding hardware normally, please report this problem to me, and provide a screenshot of LibreHardwareMonitor so that I can investigate your problem. \n\n**Note: The hardware monitoring function (including temperature monitoring and GPU usage monitoring) may have some problems, which may cause more CPU and memory usage. According to feedback from some users, turning on the temperature function will cause problems such as program crashes and system crashes. Please decide to turn on the hardware monitoring function after you are aware of the above risks. Otherwise, please do not use the hardware monitoring function.**\n\n## 14. The program prompts that \"MSVC\\*.dll\" or \"mfc\\*.dll\" cannot be found when the program starts\n\nClick the link below to download and install the Microsoft Visual C++ runtime environment.\n\n[Latest supported Visual C++ Redistributable downloads | Microsoft Docs](https://docs.microsoft.com/en-us/cpp/windows/latest-supported-vc-redist?view=msvc-170)\n\n<br><br>\n\n>If you have encountered any other problems, please click \"Contact Author\" in the \"About\" dialog box, or directly [click here](mailto:zhongyang219@hotmail.com) to send me an email.<br>\nPlease describe the problems you have encountered in detail, the error prompts, what operations you have tried, it is better to attach the screenshots and configuration files (\"Options\"-\"General Settings\"-\"Open configuration file path\"). <br>\nNote: Please confirm the email address that you used before sending the email. If your email address is form like \"Outlook_\\<random string\\>@outlook.com\", then such email address is unable to reply.\nThis may be due to the fact that you are using a third-party e-mail address as the Microsoft account to log on to Windows. If this is the case, be sure to enclose the correct email address in the message.\n"
  },
  {
    "path": "LICENSE",
    "content": "Copyright (c) <2019> <copyright zhongyang219>\n\n\"Anti 996\" License Version 1.0 (Draft)\n\nPermission is hereby granted to any individual or legal entity\nobtaining a copy of this licensed work (including the source code,\ndocumentation and/or related items, hereinafter collectively referred\nto as the \"licensed work\"), free of charge, to deal with the licensed\nwork for any purpose, including without limitation, the rights to use,\nreproduce, modify, prepare derivative works of, distribute, publish\nand sublicense the licensed work, subject to the following conditions:\n\n1. The individual or the legal entity must conspicuously display,\nwithout modification, this License and the notice on each redistributed\nor derivative copy of the Licensed Work.\n\n2. The individual or the legal entity must strictly comply with all\napplicable laws, regulations, rules and standards of the jurisdiction\nrelating to labor and employment where the individual is physically\nlocated or where the individual was born or naturalized; or where the\nlegal entity is registered or is operating (whichever is stricter). In\ncase that the jurisdiction has no such laws, regulations, rules and\nstandards or its laws, regulations, rules and standards are\nunenforceable, the individual or the legal entity are required to\ncomply with Core International Labor Standards.\n\n3. The individual or the legal entity shall not induce, suggest or force\nits employee(s), whether full-time or part-time, or its independent\ncontractor(s), in any methods, to agree in oral or written form, to\ndirectly or indirectly restrict, weaken or relinquish his or her\nrights or remedies under such laws, regulations, rules and standards\nrelating to labor and employment as mentioned above, no matter whether\nsuch written or oral agreements are enforceable under the laws of the\nsaid jurisdiction, nor shall such individual or the legal entity\nlimit, in any methods, the rights of its employee(s) or independent\ncontractor(s) from reporting or complaining to the copyright holder or\nrelevant authorities monitoring the compliance of the license about\nits violation(s) of the said license.\n\nTHE LICENSED WORK IS PROVIDED \"AS IS\", WITHOUT WARRANTY OF ANY KIND,\nEXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF\nMERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT.\nIN NO EVENT SHALL THE COPYRIGHT HOLDER BE LIABLE FOR ANY CLAIM,\nDAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR\nOTHERWISE, ARISING FROM, OUT OF OR IN ANY WAY CONNECTION WITH THE\nLICENSED WORK OR THE USE OR OTHER DEALINGS IN THE LICENSED WORK.\n"
  },
  {
    "path": "LICENSE_CN",
    "content": "版权所有（c）<2019> <copyright zhongyang219>\n\n反996许可证版本1.0\n\n在符合下列条件的情况下，特此免费向任何得到本授权作品的副本（包括源代码、文件和/或相关内容，以\n下统称为“授权作品”）的个人和法人实体授权：被授权个人或法人实体有权以任何目的处置授权作品，包括\n但不限于使用、复制，修改，衍生利用、散布，发布和再许可：\n\n1. 个人或法人实体必须在许可作品的每个再散布或衍生副本上包含以上版权声明和本许可证，不得自行修\n改。\n2. 个人或法人实体必须严格遵守与个人实际所在地或个人出生地或归化地、或法人实体注册地或经营地（\n以较严格者为准）的司法管辖区所有适用的与劳动和就业相关法律、法规、规则和标准。如果该司法管辖区\n没有此类法律、法规、规章和标准或其法律、法规、规章和标准不可执行，则个人或法人实体必须遵守国际\n劳工标准的核心公约。\n3. 个人或法人不得以任何方式诱导、暗示或强迫其全职或兼职员工或其独立承包人以口头或书面形式同意\n直接或间接限制、削弱或放弃其所拥有的，受相关与劳动和就业有关的法律、法规、规则和标准保护的权利\n或补救措施，无论该等书面或口头协议是否被该司法管辖区的法律所承认，该等个人或法人实体也不得以任\n何方法限制其雇员或独立承包人向版权持有人或监督许可证合规情况的有关当局报告或投诉上述违反许可证\n的行为的权利。\n\n该授权作品是\"按原样\"提供，不做任何明示或暗示的保证，包括但不限于对适销性、特定用途适用性和非侵\n权性的保证。在任何情况下，无论是在合同诉讼、侵权诉讼或其他诉讼中，版权持有人均不承担因本软件或\n本软件的使用或其他交易而产生、引起或与之相关的任何索赔、损害或其他责任。\n"
  },
  {
    "path": "OpenHardwareMonitorApi/LibreHardwareMonitorLib.xml",
    "content": "<?xml version=\"1.0\"?>\n<doc>\n    <assembly>\n        <name>LibreHardwareMonitorLib</name>\n    </assembly>\n    <members>\n        <member name=\"T:LibreHardwareMonitor.Hardware.Computer\">\n            <summary>\n            Stores all hardware groups and decides which devices should be enabled and updated.\n            </summary>\n        </member>\n        <member name=\"E:LibreHardwareMonitor.Hardware.Computer.HardwareAdded\">\n            <inheritdoc />\n        </member>\n        <member name=\"E:LibreHardwareMonitor.Hardware.Computer.HardwareRemoved\">\n            <inheritdoc />\n        </member>\n        <member name=\"M:LibreHardwareMonitor.Hardware.Computer.#ctor\">\n            <summary>\n            Creates a new <see cref=\"T:LibreHardwareMonitor.Hardware.IComputer\"/> instance with basic initial <see cref=\"T:LibreHardwareMonitor.Hardware.Computer.Settings\"/>.\n            </summary>\n        </member>\n        <member name=\"M:LibreHardwareMonitor.Hardware.Computer.#ctor(LibreHardwareMonitor.Hardware.ISettings)\">\n            <summary>\n            Creates a new <see cref=\"T:LibreHardwareMonitor.Hardware.IComputer\"/> instance with additional <see cref=\"T:LibreHardwareMonitor.Hardware.ISettings\"/>.\n            </summary>\n            <param name=\"settings\">Computer settings that will be transferred to each <see cref=\"T:LibreHardwareMonitor.Hardware.IHardware\"/>.</param>\n        </member>\n        <member name=\"P:LibreHardwareMonitor.Hardware.Computer.Hardware\">\n            <inheritdoc />\n        </member>\n        <member name=\"P:LibreHardwareMonitor.Hardware.Computer.SMBios\">\n            <summary>\n            Contains computer information table read in accordance with <see href=\"https://www.dmtf.org/standards/smbios\">System Management BIOS (SMBIOS) Reference Specification</see>.\n            </summary>\n        </member>\n        <member name=\"P:LibreHardwareMonitor.Hardware.Computer.IsCpuEnabled\">\n            <inheritdoc />\n        </member>\n        <member name=\"P:LibreHardwareMonitor.Hardware.Computer.IsControllerEnabled\">\n            <inheritdoc />\n        </member>\n        <member name=\"P:LibreHardwareMonitor.Hardware.Computer.IsGpuEnabled\">\n            <inheritdoc />\n        </member>\n        <member name=\"P:LibreHardwareMonitor.Hardware.Computer.IsMemoryEnabled\">\n            <inheritdoc />\n        </member>\n        <member name=\"P:LibreHardwareMonitor.Hardware.Computer.IsMotherboardEnabled\">\n            <inheritdoc />\n        </member>\n        <member name=\"P:LibreHardwareMonitor.Hardware.Computer.IsNetworkEnabled\">\n            <inheritdoc />\n        </member>\n        <member name=\"P:LibreHardwareMonitor.Hardware.Computer.IsStorageEnabled\">\n            <inheritdoc />\n        </member>\n        <member name=\"P:LibreHardwareMonitor.Hardware.Computer.IsPsuEnabled\">\n            <inheritdoc />\n        </member>\n        <member name=\"M:LibreHardwareMonitor.Hardware.Computer.Accept(LibreHardwareMonitor.Hardware.IVisitor)\">\n            <summary>\n            Triggers the <see cref=\"M:LibreHardwareMonitor.Hardware.IVisitor.VisitComputer(LibreHardwareMonitor.Hardware.IComputer)\"/> method for the given observer.\n            </summary>\n            <param name=\"visitor\">Observer who call to devices.</param>\n        </member>\n        <member name=\"M:LibreHardwareMonitor.Hardware.Computer.Traverse(LibreHardwareMonitor.Hardware.IVisitor)\">\n            <summary>\n            Triggers the <see cref=\"M:LibreHardwareMonitor.Hardware.IElement.Accept(LibreHardwareMonitor.Hardware.IVisitor)\"/> method with the given visitor for each device in each group.\n            </summary>\n            <param name=\"visitor\">Observer who call to devices.</param>\n        </member>\n        <member name=\"M:LibreHardwareMonitor.Hardware.Computer.Open\">\n            <summary>\n            If hasn't been opened before, opens <see cref=\"P:LibreHardwareMonitor.Hardware.Computer.SMBios\"/>, <see cref=\"T:LibreHardwareMonitor.Hardware.Ring0\"/>, <see cref=\"T:LibreHardwareMonitor.Hardware.OpCode\"/> and triggers the private <see cref=\"M:LibreHardwareMonitor.Hardware.Computer.AddGroups\"/> method depending on which categories are enabled.\n            </summary>\n        </member>\n        <member name=\"M:LibreHardwareMonitor.Hardware.Computer.Close\">\n            <summary>\n            If opened before, removes all <see cref=\"T:LibreHardwareMonitor.Hardware.IGroup\"/> and triggers <see cref=\"M:LibreHardwareMonitor.Hardware.OpCode.Close\"/>, <see cref=\"M:LibreHardwareMonitor.Hardware.InpOut.Close\"/> and <see cref=\"M:LibreHardwareMonitor.Hardware.Ring0.Close\"/>.\n            </summary>\n        </member>\n        <member name=\"M:LibreHardwareMonitor.Hardware.Computer.Reset\">\n            <summary>\n            If opened before, removes all <see cref=\"T:LibreHardwareMonitor.Hardware.IGroup\"/> and recreates it.\n            </summary>\n        </member>\n        <member name=\"T:LibreHardwareMonitor.Hardware.Computer.Settings\">\n            <summary>\n            <see cref=\"T:LibreHardwareMonitor.Hardware.Computer\"/> specific additional settings passed to its <see cref=\"T:LibreHardwareMonitor.Hardware.IHardware\"/>.\n            </summary>\n        </member>\n        <member name=\"T:LibreHardwareMonitor.Hardware.Controller.Nzxt.KrakenX3\">\n            Support for the Kraken X3 devices from NZXT\n        </member>\n        <member name=\"M:LibreHardwareMonitor.Hardware.CPU.CpuId.#ctor(System.Int32,System.Int32,LibreHardwareMonitor.Hardware.GroupAffinity)\">\n            <summary>\n            Initializes a new instance of the <see cref=\"T:LibreHardwareMonitor.Hardware.CPU.CpuId\" /> class.\n            </summary>\n            <param name=\"group\">The group.</param>\n            <param name=\"thread\">The thread.</param>\n            <param name=\"affinity\">The affinity.</param>\n        </member>\n        <member name=\"M:LibreHardwareMonitor.Hardware.CPU.CpuId.Get(System.Int32,System.Int32)\">\n            <summary>\n            Gets the specified <see cref=\"T:LibreHardwareMonitor.Hardware.CPU.CpuId\" />.\n            </summary>\n            <param name=\"group\">The group.</param>\n            <param name=\"thread\">The thread.</param>\n            <returns><see cref=\"T:LibreHardwareMonitor.Hardware.CPU.CpuId\" />.</returns>\n        </member>\n        <member name=\"P:LibreHardwareMonitor.Hardware.CPU.GenericCpu.CpuId\">\n            <summary>\n            Gets the CPUID.\n            </summary>\n        </member>\n        <member name=\"P:LibreHardwareMonitor.Hardware.CPU.GenericCpu.Index\">\n            <summary>\n            Gets the CPU index.\n            </summary>\n        </member>\n        <member name=\"M:LibreHardwareMonitor.Hardware.Gpu.AmdGpu.SetDefaultFanSpeed\">\n            <summary>\n            Sets the default fan speed.\n            </summary>\n        </member>\n        <member name=\"M:LibreHardwareMonitor.Hardware.Gpu.AmdGpu.GetODNTemperature(LibreHardwareMonitor.Interop.AtiAdlxx.ADLODNTemperatureType,LibreHardwareMonitor.Hardware.Sensor,System.Double,System.Double,System.Boolean)\">\n            <summary>\n            Gets the OverdriveN temperature.\n            </summary>\n            <param name=\"type\">The type.</param>\n            <param name=\"sensor\">The sensor.</param>\n            <param name=\"minTemperature\">The minimum temperature.</param>\n            <param name=\"scale\">The scale.</param>\n            <param name=\"reset\">If set to <c>true</c>, resets the sensor value to <c>null</c>.</param>\n        </member>\n        <member name=\"M:LibreHardwareMonitor.Hardware.Gpu.AmdGpu.GetPMLog(LibreHardwareMonitor.Interop.AtiAdlxx.ADLPMLogDataOutput,LibreHardwareMonitor.Interop.AtiAdlxx.ADLSensorType,LibreHardwareMonitor.Hardware.Sensor,System.Single,System.Boolean)\">\n            <summary>\n            Gets a PMLog sensor value.\n            </summary>\n            <param name=\"data\">The data.</param>\n            <param name=\"sensorType\">Type of the sensor.</param>\n            <param name=\"sensor\">The sensor.</param>\n            <param name=\"factor\">The factor.</param>\n        </member>\n        <member name=\"M:LibreHardwareMonitor.Hardware.Gpu.AmdGpu.GetOD6Power(LibreHardwareMonitor.Interop.AtiAdlxx.ADLODNCurrentPowerType,LibreHardwareMonitor.Hardware.Sensor)\">\n            <summary>\n            Gets the Overdrive6 power.\n            </summary>\n            <param name=\"type\">The type.</param>\n            <param name=\"sensor\">The sensor.</param>\n        </member>\n        <member name=\"T:LibreHardwareMonitor.Hardware.GroupAffinity\">\n            <summary>\n            This structure describes a group-specific affinity.\n            </summary>\n        </member>\n        <member name=\"M:LibreHardwareMonitor.Hardware.GroupAffinity.#ctor(System.UInt16,System.UInt64)\">\n            <summary>\n            Initializes a new instance of the <see cref=\"T:LibreHardwareMonitor.Hardware.GroupAffinity\" /> struct.\n            </summary>\n            <param name=\"group\">The group.</param>\n            <param name=\"mask\">The mask.</param>\n        </member>\n        <member name=\"M:LibreHardwareMonitor.Hardware.GroupAffinity.Single(System.UInt16,System.Int32)\">\n            <summary>\n            Gets a single group affinity.\n            </summary>\n            <param name=\"group\">The group.</param>\n            <param name=\"index\">The index.</param>\n            <returns><see cref=\"T:LibreHardwareMonitor.Hardware.GroupAffinity\" />.</returns>\n        </member>\n        <member name=\"P:LibreHardwareMonitor.Hardware.GroupAffinity.Group\">\n            <summary>\n            Gets the group.\n            </summary>\n        </member>\n        <member name=\"P:LibreHardwareMonitor.Hardware.GroupAffinity.Mask\">\n            <summary>\n            Gets the mask.\n            </summary>\n        </member>\n        <member name=\"M:LibreHardwareMonitor.Hardware.GroupAffinity.Equals(System.Object)\">\n            <summary>\n            Determines whether the specified <see cref=\"T:System.Object\" /> is equal to this instance.\n            </summary>\n            <param name=\"o\">The <see cref=\"T:System.Object\" /> to compare with this instance.</param>\n            <returns><c>true</c> if the specified <see cref=\"T:System.Object\" /> is equal to this instance; otherwise, <c>false</c>.</returns>\n        </member>\n        <member name=\"M:LibreHardwareMonitor.Hardware.GroupAffinity.GetHashCode\">\n            <summary>\n            Returns a hash code for this instance.\n            </summary>\n            <returns>A hash code for this instance, suitable for use in hashing algorithms and data structures like a hash table.</returns>\n        </member>\n        <member name=\"M:LibreHardwareMonitor.Hardware.GroupAffinity.op_Equality(LibreHardwareMonitor.Hardware.GroupAffinity,LibreHardwareMonitor.Hardware.GroupAffinity)\">\n            <summary>\n            Implements the == operator.\n            </summary>\n            <param name=\"a1\">The a1.</param>\n            <param name=\"a2\">The a2.</param>\n            <returns>The result of the operator.</returns>\n        </member>\n        <member name=\"M:LibreHardwareMonitor.Hardware.GroupAffinity.op_Inequality(LibreHardwareMonitor.Hardware.GroupAffinity,LibreHardwareMonitor.Hardware.GroupAffinity)\">\n            <summary>\n            Implements the != operator.\n            </summary>\n            <param name=\"a1\">The a1.</param>\n            <param name=\"a2\">The a2.</param>\n            <returns>The result of the operator.</returns>\n        </member>\n        <member name=\"T:LibreHardwareMonitor.Hardware.Hardware\">\n            <summary>\n            Object representing a component of the computer.\n            <para>\n            Individual information can be read from the <see cref=\"P:LibreHardwareMonitor.Hardware.Hardware.Sensors\"/>.\n            </para>\n            </summary>\n        </member>\n        <member name=\"M:LibreHardwareMonitor.Hardware.Hardware.#ctor(System.String,LibreHardwareMonitor.Hardware.Identifier,LibreHardwareMonitor.Hardware.ISettings)\">\n            <summary>\n            Creates a new <see cref=\"T:LibreHardwareMonitor.Hardware.Hardware\"/> instance based on the data provided.\n            </summary>\n            <param name=\"name\">Component name.</param>\n            <param name=\"identifier\">Identifier that will be assigned to the device. Based on <see cref=\"P:LibreHardwareMonitor.Hardware.Hardware.Identifier\"/></param>\n            <param name=\"settings\">Additional settings passed by the <see cref=\"T:LibreHardwareMonitor.Hardware.IComputer\"/>.</param>\n        </member>\n        <member name=\"E:LibreHardwareMonitor.Hardware.Hardware.Closing\">\n            <summary>\n            Event triggered when <see cref=\"T:LibreHardwareMonitor.Hardware.Hardware\"/> is closing.\n            </summary>\n        </member>\n        <member name=\"P:LibreHardwareMonitor.Hardware.Hardware.HardwareType\">\n            <inheritdoc />\n        </member>\n        <member name=\"P:LibreHardwareMonitor.Hardware.Hardware.Identifier\">\n            <inheritdoc />\n        </member>\n        <member name=\"P:LibreHardwareMonitor.Hardware.Hardware.Name\">\n            <inheritdoc />\n        </member>\n        <member name=\"P:LibreHardwareMonitor.Hardware.Hardware.Parent\">\n            <inheritdoc />\n        </member>\n        <member name=\"P:LibreHardwareMonitor.Hardware.Hardware.Properties\">\n            <inheritdoc />\n        </member>\n        <member name=\"P:LibreHardwareMonitor.Hardware.Hardware.Sensors\">\n            <inheritdoc />\n        </member>\n        <member name=\"P:LibreHardwareMonitor.Hardware.Hardware.SubHardware\">\n            <inheritdoc />\n        </member>\n        <member name=\"M:LibreHardwareMonitor.Hardware.Hardware.GetReport\">\n            <inheritdoc />\n        </member>\n        <member name=\"M:LibreHardwareMonitor.Hardware.Hardware.Update\">\n            <inheritdoc />\n        </member>\n        <member name=\"M:LibreHardwareMonitor.Hardware.Hardware.Accept(LibreHardwareMonitor.Hardware.IVisitor)\">\n            <inheritdoc />\n        </member>\n        <member name=\"M:LibreHardwareMonitor.Hardware.Hardware.Traverse(LibreHardwareMonitor.Hardware.IVisitor)\">\n            <inheritdoc />\n        </member>\n        <member name=\"M:LibreHardwareMonitor.Hardware.Hardware.ActivateSensor(LibreHardwareMonitor.Hardware.ISensor)\">\n            <inheritdoc />\n        </member>\n        <member name=\"M:LibreHardwareMonitor.Hardware.Hardware.DeactivateSensor(LibreHardwareMonitor.Hardware.ISensor)\">\n            <inheritdoc />\n        </member>\n        <member name=\"M:LibreHardwareMonitor.Hardware.Hardware.Close\">\n            <inheritdoc />\n        </member>\n        <member name=\"T:LibreHardwareMonitor.Hardware.HardwareType\">\n            <summary>\n            Collection of identifiers representing the purpose of the hardware.\n            </summary>\n        </member>\n        <member name=\"T:LibreHardwareMonitor.Hardware.HardwareEventHandler\">\n            <summary>\n            Handler that will trigger the actions assigned to it when the event occurs.\n            </summary>\n            <param name=\"hardware\">Component returned to the assigned action(s).</param>\n        </member>\n        <member name=\"T:LibreHardwareMonitor.Hardware.IComputer\">\n            <summary>\n            Basic abstract with methods for the class which can store all hardware and decides which devices are to be checked and updated.\n            </summary>\n        </member>\n        <member name=\"P:LibreHardwareMonitor.Hardware.IComputer.IsCpuEnabled\">\n            <summary>\n            Gets or sets a value indicating whether collecting information about <see cref=\"F:LibreHardwareMonitor.Hardware.HardwareType.Cpu\"/> devices should be enabled and updated.\n            </summary>\n            <returns><see langword=\"true\"/> if a given category of devices is already enabled.</returns>\n        </member>\n        <member name=\"P:LibreHardwareMonitor.Hardware.IComputer.IsControllerEnabled\">\n            <summary>\n            Gets or sets a value indicating whether collecting information about:\n            <list>\n            <item><see cref=\"T:LibreHardwareMonitor.Hardware.Controller.TBalancer.TBalancerGroup\"/></item>\n            <item><see cref=\"T:LibreHardwareMonitor.Hardware.Controller.Heatmaster.HeatmasterGroup\"/></item>\n            <item><see cref=\"T:LibreHardwareMonitor.Hardware.Controller.AquaComputer.AquaComputerGroup\"/></item>\n            <item><see cref=\"T:LibreHardwareMonitor.Hardware.Controller.AeroCool.AeroCoolGroup\"/></item>\n            <item><see cref=\"T:LibreHardwareMonitor.Hardware.Controller.Nzxt.NzxtGroup\"/></item>\n            </list>\n            devices should be enabled and updated.\n            </summary>\n            <returns><see langword=\"true\"/> if a given category of devices is already enabled.</returns>\n        </member>\n        <member name=\"P:LibreHardwareMonitor.Hardware.IComputer.IsGpuEnabled\">\n            <summary>\n            Gets or sets a value indicating whether collecting information about <see cref=\"F:LibreHardwareMonitor.Hardware.HardwareType.GpuAmd\"/> or <see cref=\"F:LibreHardwareMonitor.Hardware.HardwareType.GpuNvidia\"/> devices should be enabled and updated.\n            </summary>\n            <returns><see langword=\"true\"/> if a given category of devices is already enabled.</returns>\n        </member>\n        <member name=\"P:LibreHardwareMonitor.Hardware.IComputer.IsStorageEnabled\">\n            <summary>\n            Gets or sets a value indicating whether collecting information about <see cref=\"F:LibreHardwareMonitor.Hardware.HardwareType.Storage\"/> devices should be enabled and updated.\n            </summary>\n            <returns><see langword=\"true\"/> if a given category of devices is already enabled.</returns>\n        </member>\n        <member name=\"P:LibreHardwareMonitor.Hardware.IComputer.IsMotherboardEnabled\">\n            <summary>\n            Gets or sets a value indicating whether collecting information about <see cref=\"F:LibreHardwareMonitor.Hardware.HardwareType.Motherboard\"/> devices should be enabled and updated.\n            </summary>\n            <returns><see langword=\"true\"/> if a given category of devices is already enabled.</returns>\n        </member>\n        <member name=\"P:LibreHardwareMonitor.Hardware.IComputer.IsNetworkEnabled\">\n            <summary>\n            Gets or sets a value indicating whether collecting information about <see cref=\"F:LibreHardwareMonitor.Hardware.HardwareType.Network\"/> devices should be enabled and updated.\n            </summary>\n            <returns><see langword=\"true\"/> if a given category of devices is already enabled.</returns>\n        </member>\n        <member name=\"P:LibreHardwareMonitor.Hardware.IComputer.IsMemoryEnabled\">\n            <summary>\n            Gets or sets a value indicating whether collecting information about <see cref=\"F:LibreHardwareMonitor.Hardware.HardwareType.Memory\"/> devices should be enabled and updated.\n            </summary>\n            <returns><see langword=\"true\"/> if a given category of devices is already enabled.</returns>\n        </member>\n        <member name=\"P:LibreHardwareMonitor.Hardware.IComputer.IsPsuEnabled\">\n            <summary>\n            Gets or sets a value indicating whether collecting information about <see cref=\"F:LibreHardwareMonitor.Hardware.HardwareType.Psu\"/> devices should be enabled and updated.\n            </summary>\n            <returns><see langword=\"true\"/> if a given category of devices is already enabled.</returns>\n        </member>\n        <member name=\"P:LibreHardwareMonitor.Hardware.IComputer.Hardware\">\n            <summary>\n            Gets a list of all known <see cref=\"T:LibreHardwareMonitor.Hardware.IHardware\"/>.\n            <para>Can be updated by <see cref=\"T:LibreHardwareMonitor.Hardware.IVisitor\"/>.</para>\n            </summary>\n            <returns>List of all enabled devices.</returns>\n        </member>\n        <member name=\"E:LibreHardwareMonitor.Hardware.IComputer.HardwareAdded\">\n            <summary>\n            Triggered when a new <see cref=\"T:LibreHardwareMonitor.Hardware.IHardware\"/> is registered.\n            </summary>\n        </member>\n        <member name=\"E:LibreHardwareMonitor.Hardware.IComputer.HardwareRemoved\">\n            <summary>\n            Triggered when a <see cref=\"T:LibreHardwareMonitor.Hardware.IHardware\"/> is removed.\n            </summary>\n        </member>\n        <member name=\"M:LibreHardwareMonitor.Hardware.IComputer.GetReport\">\n            <summary>\n            Generates full LibreHardwareMonitor report for devices that have been enabled.\n            </summary>\n            <returns>A formatted text string with library, OS and hardware information.</returns>\n        </member>\n        <member name=\"T:LibreHardwareMonitor.Hardware.Identifier\">\n            <summary>\n            Represents a unique <see cref=\"T:LibreHardwareMonitor.Hardware.ISensor\"/>/<see cref=\"T:LibreHardwareMonitor.Hardware.IHardware\"/> identifier in text format with a / separator.\n            </summary>\n        </member>\n        <member name=\"M:LibreHardwareMonitor.Hardware.Identifier.#ctor(LibreHardwareMonitor.Hardware.Identifier,System.String[])\">\n            <summary>\n            Creates a new identifier instance based on the base <see cref=\"T:LibreHardwareMonitor.Hardware.Identifier\"/> and additional elements.\n            </summary>\n            <param name=\"identifier\">Base identifier being the beginning of the new one.</param>\n            <param name=\"extensions\">Additional parts by which the base <see cref=\"T:LibreHardwareMonitor.Hardware.Identifier\"/> will be extended.</param>\n        </member>\n        <member name=\"T:LibreHardwareMonitor.Hardware.IElement\">\n            <summary>\n            Abstract parent with logic for the abstract class that stores data.\n            </summary>\n        </member>\n        <member name=\"M:LibreHardwareMonitor.Hardware.IElement.Accept(LibreHardwareMonitor.Hardware.IVisitor)\">\n            <summary>\n            Accepts the observer for this instance.\n            </summary>\n            <param name=\"visitor\">Computer observer making the calls.</param>\n        </member>\n        <member name=\"M:LibreHardwareMonitor.Hardware.IElement.Traverse(LibreHardwareMonitor.Hardware.IVisitor)\">\n            <summary>\n            Call the <see cref=\"M:LibreHardwareMonitor.Hardware.IElement.Accept(LibreHardwareMonitor.Hardware.IVisitor)\"/> method for all child instances <c>(called only from visitors).</c>\n            </summary>\n            <param name=\"visitor\">Computer observer making the calls.</param>\n        </member>\n        <member name=\"T:LibreHardwareMonitor.Hardware.IGroup\">\n            <summary>\n            A group of devices from one category in one list.\n            </summary>\n        </member>\n        <member name=\"P:LibreHardwareMonitor.Hardware.IGroup.Hardware\">\n            <summary>\n            Gets a list that stores information about <see cref=\"T:LibreHardwareMonitor.Hardware.IHardware\"/> in a given group.\n            </summary>\n        </member>\n        <member name=\"M:LibreHardwareMonitor.Hardware.IGroup.GetReport\">\n            <summary>\n            Report containing most of the known information about all <see cref=\"T:LibreHardwareMonitor.Hardware.IHardware\"/> in this <see cref=\"T:LibreHardwareMonitor.Hardware.IGroup\"/>.\n            </summary>\n            <returns>A formatted text string with hardware information.</returns>\n        </member>\n        <member name=\"M:LibreHardwareMonitor.Hardware.IGroup.Close\">\n            <summary>\n            Stop updating this group in the future.\n            </summary>\n        </member>\n        <member name=\"T:LibreHardwareMonitor.Hardware.SensorEventHandler\">\n            <summary>\n            Handler that will trigger the actions assigned to it when the event occurs.\n            </summary>\n            <param name=\"sensor\">Component returned to the assigned action(s).</param>\n        </member>\n        <member name=\"T:LibreHardwareMonitor.Hardware.IHardware\">\n            <summary>\n            Abstract object that stores information about a device. All sensors are available as an array of <see cref=\"P:LibreHardwareMonitor.Hardware.IHardware.Sensors\"/>.\n            <para>\n            Can contain <see cref=\"P:LibreHardwareMonitor.Hardware.IHardware.SubHardware\"/>.\n            Type specified in <see cref=\"P:LibreHardwareMonitor.Hardware.IHardware.HardwareType\"/>.\n            </para>\n            </summary>\n        </member>\n        <member name=\"P:LibreHardwareMonitor.Hardware.IHardware.HardwareType\">\n            <summary>\n            <inheritdoc cref=\"T:LibreHardwareMonitor.Hardware.HardwareType\"/>\n            </summary>\n        </member>\n        <member name=\"P:LibreHardwareMonitor.Hardware.IHardware.Identifier\">\n            <summary>\n            Gets a unique hardware ID that represents its location.\n            </summary>\n        </member>\n        <member name=\"P:LibreHardwareMonitor.Hardware.IHardware.Name\">\n            <summary>\n            Gets or sets device name.\n            </summary>\n        </member>\n        <member name=\"P:LibreHardwareMonitor.Hardware.IHardware.Parent\">\n            <summary>\n            Gets the device that is the parent of the current hardware. For example, the motherboard is the parent of SuperIO.\n            </summary>\n        </member>\n        <member name=\"P:LibreHardwareMonitor.Hardware.IHardware.Sensors\">\n            <summary>\n            Gets an array of all sensors such as <see cref=\"F:LibreHardwareMonitor.Hardware.SensorType.Temperature\"/>, <see cref=\"F:LibreHardwareMonitor.Hardware.SensorType.Clock\"/>, <see cref=\"F:LibreHardwareMonitor.Hardware.SensorType.Load\"/> etc.\n            </summary>\n        </member>\n        <member name=\"P:LibreHardwareMonitor.Hardware.IHardware.SubHardware\">\n            <summary>\n            Gets child devices, e.g. <see cref=\"T:LibreHardwareMonitor.Hardware.Motherboard.Lpc.LpcIO\"/> of the <see cref=\"T:LibreHardwareMonitor.Hardware.Motherboard.Motherboard\"/>.\n            </summary>\n        </member>\n        <member name=\"M:LibreHardwareMonitor.Hardware.IHardware.GetReport\">\n            <summary>\n            Report containing most of the known information about the current device.\n            </summary>\n            <returns>A formatted text string with hardware information.</returns>\n        </member>\n        <member name=\"M:LibreHardwareMonitor.Hardware.IHardware.Update\">\n            <summary>\n            Refreshes the information stored in <see cref=\"P:LibreHardwareMonitor.Hardware.IHardware.Sensors\"/> array.\n            </summary>\n        </member>\n        <member name=\"E:LibreHardwareMonitor.Hardware.IHardware.SensorAdded\">\n            <summary>\n            An <see langword=\"event\"/> that will be triggered when a new sensor appears.\n            </summary>\n        </member>\n        <member name=\"E:LibreHardwareMonitor.Hardware.IHardware.SensorRemoved\">\n            <summary>\n            An <see langword=\"event\"/> that will be triggered when one of the sensors is removed.\n            \n            </summary>\n        </member>\n        <member name=\"P:LibreHardwareMonitor.Hardware.IHardware.Properties\">\n            <summary>\n            Gets rarely changed hardware properties that can't be represented as sensors.\n            </summary>\n        </member>\n        <member name=\"T:LibreHardwareMonitor.Hardware.IParameter\">\n            <summary>\n            Abstract object that represents additional parameters included in <see cref=\"T:LibreHardwareMonitor.Hardware.ISensor\"/>.\n            </summary>\n        </member>\n        <member name=\"P:LibreHardwareMonitor.Hardware.IParameter.DefaultValue\">\n            <summary>\n            Gets a parameter default value defined by library.\n            </summary>\n        </member>\n        <member name=\"P:LibreHardwareMonitor.Hardware.IParameter.Description\">\n            <summary>\n            Gets a parameter description defined by library.\n            </summary>\n        </member>\n        <member name=\"P:LibreHardwareMonitor.Hardware.IParameter.Identifier\">\n            <summary>\n            Gets a unique parameter ID that represents its location.\n            </summary>\n        </member>\n        <member name=\"P:LibreHardwareMonitor.Hardware.IParameter.IsDefault\">\n            <summary>\n            Gets or sets information whether the given <see cref=\"T:LibreHardwareMonitor.Hardware.IParameter\"/> is the default for <see cref=\"T:LibreHardwareMonitor.Hardware.ISensor\"/>.\n            </summary>\n        </member>\n        <member name=\"P:LibreHardwareMonitor.Hardware.IParameter.Name\">\n            <summary>\n            Gets a parameter name defined by library.\n            </summary>\n        </member>\n        <member name=\"P:LibreHardwareMonitor.Hardware.IParameter.Sensor\">\n            <summary>\n            Gets the sensor that is the data container for the given parameter.\n            </summary>\n        </member>\n        <member name=\"P:LibreHardwareMonitor.Hardware.IParameter.Value\">\n            <summary>\n            Gets or sets the current value.\n            </summary>\n        </member>\n        <member name=\"T:LibreHardwareMonitor.Hardware.SensorType\">\n            <summary>\n            Category of what type the selected sensor is.\n            </summary>\n        </member>\n        <member name=\"T:LibreHardwareMonitor.Hardware.SensorValue\">\n            <summary>\n            Stores the readed value and the time in which it was recorded.\n            </summary>\n        </member>\n        <member name=\"M:LibreHardwareMonitor.Hardware.SensorValue.#ctor(System.Single,System.DateTime)\">\n            <param name=\"value\"><see cref=\"P:LibreHardwareMonitor.Hardware.SensorValue.Value\"/> of the sensor.</param>\n            <param name=\"time\">The time code during which the <see cref=\"P:LibreHardwareMonitor.Hardware.SensorValue.Value\"/> was recorded.</param>\n        </member>\n        <member name=\"P:LibreHardwareMonitor.Hardware.SensorValue.Value\">\n            <summary>\n            Gets the value of the sensor\n            </summary>\n        </member>\n        <member name=\"P:LibreHardwareMonitor.Hardware.SensorValue.Time\">\n            <summary>\n            Gets the time code during which the <see cref=\"P:LibreHardwareMonitor.Hardware.SensorValue.Value\"/> was recorded.\n            </summary>\n        </member>\n        <member name=\"T:LibreHardwareMonitor.Hardware.ISensor\">\n            <summary>\n            Stores information about the readed values and the time in which they were collected.\n            </summary>\n        </member>\n        <member name=\"P:LibreHardwareMonitor.Hardware.ISensor.Hardware\">\n            <summary>\n            <inheritdoc cref=\"T:LibreHardwareMonitor.Hardware.IHardware\"/>\n            </summary>\n        </member>\n        <member name=\"P:LibreHardwareMonitor.Hardware.ISensor.Index\">\n            <summary>\n            Gets the unique identifier of this sensor for a given <see cref=\"T:LibreHardwareMonitor.Hardware.IHardware\"/>.\n            </summary>\n        </member>\n        <member name=\"P:LibreHardwareMonitor.Hardware.ISensor.Max\">\n            <summary>\n            Gets a maximum value recorded for the given sensor.\n            </summary>\n        </member>\n        <member name=\"P:LibreHardwareMonitor.Hardware.ISensor.Min\">\n            <summary>\n            Gets a minimum value recorded for the given sensor.\n            </summary>\n        </member>\n        <member name=\"P:LibreHardwareMonitor.Hardware.ISensor.Name\">\n            <summary>\n            Gets or sets a sensor name.\n            <para>By default determined by the library.</para>\n            </summary>\n        </member>\n        <member name=\"P:LibreHardwareMonitor.Hardware.ISensor.SensorType\">\n            <summary>\n            <inheritdoc cref=\"T:LibreHardwareMonitor.Hardware.SensorType\"/>\n            </summary>\n        </member>\n        <member name=\"P:LibreHardwareMonitor.Hardware.ISensor.Value\">\n            <summary>\n            Gets the last recorded value for the given sensor.\n            </summary>\n        </member>\n        <member name=\"P:LibreHardwareMonitor.Hardware.ISensor.Values\">\n            <summary>\n            Gets a list of recorded values for the given sensor.\n            </summary>\n        </member>\n        <member name=\"M:LibreHardwareMonitor.Hardware.ISensor.ResetMin\">\n            <summary>\n            Resets a value stored in <see cref=\"P:LibreHardwareMonitor.Hardware.ISensor.Min\"/>.\n            </summary>\n        </member>\n        <member name=\"M:LibreHardwareMonitor.Hardware.ISensor.ResetMax\">\n            <summary>\n            Resets a value stored in <see cref=\"P:LibreHardwareMonitor.Hardware.ISensor.Max\"/>.\n            </summary>\n        </member>\n        <member name=\"T:LibreHardwareMonitor.Hardware.ISensorLimits\">\n            <summary>\n            Abstract object that stores information about the limits of <see cref=\"T:LibreHardwareMonitor.Hardware.ISensor\"/>.\n            </summary>\n        </member>\n        <member name=\"P:LibreHardwareMonitor.Hardware.ISensorLimits.HighLimit\">\n            <summary>\n            Upper limit of <see cref=\"T:LibreHardwareMonitor.Hardware.ISensor\"/> value.\n            </summary>\n        </member>\n        <member name=\"P:LibreHardwareMonitor.Hardware.ISensorLimits.LowLimit\">\n            <summary>\n            Lower limit of <see cref=\"T:LibreHardwareMonitor.Hardware.ISensor\"/> value.\n            </summary>\n        </member>\n        <member name=\"T:LibreHardwareMonitor.Hardware.ICriticalSensorLimits\">\n            <summary>\n            Abstract object that stores information about the critical limits of <see cref=\"T:LibreHardwareMonitor.Hardware.ISensor\"/>.\n            </summary>\n        </member>\n        <member name=\"P:LibreHardwareMonitor.Hardware.ICriticalSensorLimits.CriticalHighLimit\">\n            <summary>\n            Critical upper limit of <see cref=\"T:LibreHardwareMonitor.Hardware.ISensor\"/> value.\n            </summary>\n        </member>\n        <member name=\"P:LibreHardwareMonitor.Hardware.ICriticalSensorLimits.CriticalLowLimit\">\n            <summary>\n            Critical lower limit of <see cref=\"T:LibreHardwareMonitor.Hardware.ISensor\"/> value.\n            </summary>\n        </member>\n        <member name=\"T:LibreHardwareMonitor.Hardware.ISettings\">\n            <summary>\n            Abstract object that stores settings passed to <see cref=\"T:LibreHardwareMonitor.Hardware.IComputer\"/>, <see cref=\"T:LibreHardwareMonitor.Hardware.IHardware\"/> and <see cref=\"T:LibreHardwareMonitor.Hardware.ISensor\"/>.\n            </summary>\n        </member>\n        <member name=\"M:LibreHardwareMonitor.Hardware.ISettings.Contains(System.String)\">\n            <summary>\n            Returns information whether the given collection of settings contains a value assigned to the given key.\n            </summary>\n            <param name=\"name\">Key to which the setting value is assigned.</param>\n        </member>\n        <member name=\"M:LibreHardwareMonitor.Hardware.ISettings.SetValue(System.String,System.String)\">\n            <summary>\n            Assigns a setting option to a given key.\n            </summary>\n            <param name=\"name\">Key to which the setting value is assigned.</param>\n            <param name=\"value\">Text setting value.</param>\n        </member>\n        <member name=\"M:LibreHardwareMonitor.Hardware.ISettings.GetValue(System.String,System.String)\">\n            <summary>\n            Gets a setting option assigned to the given key.\n            </summary>\n            <param name=\"name\">Key to which the setting value is assigned.</param>\n            <param name=\"value\">Default value.</param>\n        </member>\n        <member name=\"M:LibreHardwareMonitor.Hardware.ISettings.Remove(System.String)\">\n            <summary>\n            Removes a setting with the specified key from the settings collection.\n            </summary>\n            <param name=\"name\">Key to which the setting value is assigned.</param>\n        </member>\n        <member name=\"T:LibreHardwareMonitor.Hardware.IVisitor\">\n            <summary>\n            Base interface for creating observers who call to devices.\n            </summary>\n        </member>\n        <member name=\"M:LibreHardwareMonitor.Hardware.IVisitor.VisitComputer(LibreHardwareMonitor.Hardware.IComputer)\">\n            <summary>\n            Refreshes the values of all <see cref=\"T:LibreHardwareMonitor.Hardware.ISensor\"/> in all <see cref=\"T:LibreHardwareMonitor.Hardware.IHardware\"/> on selected <see cref=\"T:LibreHardwareMonitor.Hardware.IComputer\"/>.\n            </summary>\n            <param name=\"computer\">Instance of the computer to be revisited.</param>\n        </member>\n        <member name=\"M:LibreHardwareMonitor.Hardware.IVisitor.VisitHardware(LibreHardwareMonitor.Hardware.IHardware)\">\n            <summary>\n            Refreshes the values of all <see cref=\"T:LibreHardwareMonitor.Hardware.ISensor\"/> on selected <see cref=\"T:LibreHardwareMonitor.Hardware.IHardware\"/>.\n            </summary>\n            <param name=\"hardware\">Instance of the hardware to be revisited.</param>\n        </member>\n        <member name=\"M:LibreHardwareMonitor.Hardware.IVisitor.VisitSensor(LibreHardwareMonitor.Hardware.ISensor)\">\n            <summary>\n            Refreshes the values on selected <see cref=\"T:LibreHardwareMonitor.Hardware.ISensor\"/>.\n            </summary>\n            <param name=\"sensor\">Instance of the sensor to be revisited.</param>\n        </member>\n        <member name=\"M:LibreHardwareMonitor.Hardware.IVisitor.VisitParameter(LibreHardwareMonitor.Hardware.IParameter)\">\n            <summary>\n            Refreshes the values on selected <see cref=\"T:LibreHardwareMonitor.Hardware.IParameter\"/>.\n            </summary>\n            <param name=\"parameter\">Instance of the parameter to be revisited.</param>\n        </member>\n        <member name=\"T:LibreHardwareMonitor.Hardware.Motherboard.Lpc.EC.WindowsEmbeddedControllerIO\">\n            <summary>\n            An unsafe but universal implementation for the ACPI Embedded Controller IO interface for Windows\n            </summary>\n            <remarks>\n            It is unsafe because of possible race condition between this application and the PC firmware when\n            writing to the EC registers. For a safe approach ACPI/WMI methods have to be used, but those are\n            different for each motherboard model.\n            </remarks>\n        </member>\n        <member name=\"T:LibreHardwareMonitor.Hardware.Motherboard.Motherboard\">\n            <summary>\n            Represents the motherboard of a computer with its <see cref=\"T:LibreHardwareMonitor.Hardware.Motherboard.Lpc.LpcIO\"/> and <see cref=\"T:LibreHardwareMonitor.Hardware.Motherboard.Lpc.EC.EmbeddedController\"/> as <see cref=\"P:LibreHardwareMonitor.Hardware.Motherboard.Motherboard.SubHardware\"/>.\n            </summary>\n        </member>\n        <member name=\"M:LibreHardwareMonitor.Hardware.Motherboard.Motherboard.#ctor(LibreHardwareMonitor.Hardware.SMBios,LibreHardwareMonitor.Hardware.ISettings)\">\n            <summary>\n            Creates motherboard instance by retrieving information from <see cref=\"T:LibreHardwareMonitor.Hardware.SMBios\"/> and creates a new <see cref=\"P:LibreHardwareMonitor.Hardware.Motherboard.Motherboard.SubHardware\"/> based on data from <see cref=\"T:LibreHardwareMonitor.Hardware.Motherboard.Lpc.LpcIO\"/> and <see cref=\"T:LibreHardwareMonitor.Hardware.Motherboard.Lpc.EC.EmbeddedController\"/>.\n            </summary>\n            <param name=\"smBios\"><see cref=\"T:LibreHardwareMonitor.Hardware.SMBios\"/> table containing motherboard data.</param>\n            <param name=\"settings\">Additional settings passed by <see cref=\"T:LibreHardwareMonitor.Hardware.IComputer\"/>.</param>\n        </member>\n        <member name=\"E:LibreHardwareMonitor.Hardware.Motherboard.Motherboard.SensorAdded\">\n            <inheritdoc/>\n        </member>\n        <member name=\"E:LibreHardwareMonitor.Hardware.Motherboard.Motherboard.SensorRemoved\">\n            <inheritdoc/>\n        </member>\n        <member name=\"P:LibreHardwareMonitor.Hardware.Motherboard.Motherboard.HardwareType\">\n            <returns><see cref=\"F:LibreHardwareMonitor.Hardware.HardwareType.Motherboard\"/></returns>\n        </member>\n        <member name=\"P:LibreHardwareMonitor.Hardware.Motherboard.Motherboard.Identifier\">\n            <inheritdoc/>\n        </member>\n        <member name=\"P:LibreHardwareMonitor.Hardware.Motherboard.Motherboard.Name\">\n            <summary>\n            Gets the name obtained from <see cref=\"T:LibreHardwareMonitor.Hardware.SMBios\"/>.\n            </summary>\n        </member>\n        <member name=\"P:LibreHardwareMonitor.Hardware.Motherboard.Motherboard.Parent\">\n            <inheritdoc/>\n            <returns>Always <see langword=\"null\"/></returns>\n        </member>\n        <member name=\"P:LibreHardwareMonitor.Hardware.Motherboard.Motherboard.Properties\">\n            <inheritdoc/>\n        </member>\n        <member name=\"P:LibreHardwareMonitor.Hardware.Motherboard.Motherboard.Sensors\">\n            <inheritdoc/>\n        </member>\n        <member name=\"P:LibreHardwareMonitor.Hardware.Motherboard.Motherboard.SMBios\">\n            <summary>\n            Gets the <see cref=\"T:LibreHardwareMonitor.Hardware.SMBios\"/> information.\n            </summary>\n        </member>\n        <member name=\"P:LibreHardwareMonitor.Hardware.Motherboard.Motherboard.SubHardware\">\n            <inheritdoc/>\n        </member>\n        <member name=\"M:LibreHardwareMonitor.Hardware.Motherboard.Motherboard.GetReport\">\n            <inheritdoc/>\n        </member>\n        <member name=\"M:LibreHardwareMonitor.Hardware.Motherboard.Motherboard.Update\">\n            <summary>\n            Motherboard itself cannot be updated. Update <see cref=\"P:LibreHardwareMonitor.Hardware.Motherboard.Motherboard.SubHardware\"/> instead.\n            </summary>\n        </member>\n        <member name=\"M:LibreHardwareMonitor.Hardware.Motherboard.Motherboard.Accept(LibreHardwareMonitor.Hardware.IVisitor)\">\n            <inheritdoc/>\n        </member>\n        <member name=\"M:LibreHardwareMonitor.Hardware.Motherboard.Motherboard.Traverse(LibreHardwareMonitor.Hardware.IVisitor)\">\n            <inheritdoc/>\n        </member>\n        <member name=\"M:LibreHardwareMonitor.Hardware.Motherboard.Motherboard.Close\">\n            <summary>\n            Closes <see cref=\"P:LibreHardwareMonitor.Hardware.Motherboard.Motherboard.SubHardware\"/> using <see cref=\"M:LibreHardwareMonitor.Hardware.Hardware.Close\"/>.\n            </summary>\n        </member>\n        <member name=\"T:LibreHardwareMonitor.Hardware.ParameterDescription\">\n            <summary>\n            Composite class containing information about the selected <see cref=\"T:LibreHardwareMonitor.Hardware.ISensor\"/>.\n            </summary>\n        </member>\n        <member name=\"M:LibreHardwareMonitor.Hardware.ParameterDescription.#ctor(System.String,System.String,System.Single)\">\n            <summary>\n            Creates a new instance and assigns values.\n            </summary>\n            <param name=\"name\">Name of the selected component.</param>\n            <param name=\"description\">Description of the selected component.</param>\n            <param name=\"defaultValue\">Default value of the selected component.</param>\n        </member>\n        <member name=\"P:LibreHardwareMonitor.Hardware.ParameterDescription.Name\">\n            <summary>\n            Gets a name of the parent <see cref=\"T:LibreHardwareMonitor.Hardware.ISensor\"/>.\n            </summary>\n        </member>\n        <member name=\"P:LibreHardwareMonitor.Hardware.ParameterDescription.Description\">\n            <summary>\n            Gets a description of the parent <see cref=\"T:LibreHardwareMonitor.Hardware.ISensor\"/>.\n            </summary>\n        </member>\n        <member name=\"P:LibreHardwareMonitor.Hardware.ParameterDescription.DefaultValue\">\n            <summary>\n            Gets a default value of the parent <see cref=\"T:LibreHardwareMonitor.Hardware.ISensor\"/>.\n            </summary>\n        </member>\n        <member name=\"T:LibreHardwareMonitor.Hardware.SensorVisitor\">\n            <summary>\n            Observer making calls to selected component <see cref=\"T:LibreHardwareMonitor.Hardware.ISensor\"/>'s.\n            </summary>\n        </member>\n        <member name=\"M:LibreHardwareMonitor.Hardware.SensorVisitor.#ctor(LibreHardwareMonitor.Hardware.SensorEventHandler)\">\n            <summary>\n            Creates a new observer instance.\n            </summary>\n            <param name=\"handler\">Instance of the <see cref=\"T:LibreHardwareMonitor.Hardware.SensorEventHandler\"/> that triggers events during visiting the <see cref=\"T:LibreHardwareMonitor.Hardware.ISensor\"/>.</param>\n        </member>\n        <member name=\"M:LibreHardwareMonitor.Hardware.SensorVisitor.VisitComputer(LibreHardwareMonitor.Hardware.IComputer)\">\n            <summary>\n            Goes through all the components of the specified <see cref=\"T:LibreHardwareMonitor.Hardware.IComputer\"/> with its <see cref=\"M:LibreHardwareMonitor.Hardware.IElement.Traverse(LibreHardwareMonitor.Hardware.IVisitor)\"/>.\n            </summary>\n            <param name=\"computer\">Computer class instance that is derived from the <see cref=\"T:LibreHardwareMonitor.Hardware.IComputer\"/> interface.</param>\n        </member>\n        <member name=\"M:LibreHardwareMonitor.Hardware.SensorVisitor.VisitHardware(LibreHardwareMonitor.Hardware.IHardware)\">\n            <summary>\n            Goes through all the components of the specified <see cref=\"T:LibreHardwareMonitor.Hardware.IHardware\"/> with its <see cref=\"M:LibreHardwareMonitor.Hardware.IElement.Traverse(LibreHardwareMonitor.Hardware.IVisitor)\"/>.\n            </summary>\n            <param name=\"hardware\">Hardware class instance that is derived from the <see cref=\"T:LibreHardwareMonitor.Hardware.IHardware\"/> interface.</param>\n        </member>\n        <member name=\"M:LibreHardwareMonitor.Hardware.SensorVisitor.VisitSensor(LibreHardwareMonitor.Hardware.ISensor)\">\n            <summary>\n            Goes through all the components of the specified <see cref=\"T:LibreHardwareMonitor.Hardware.ISensor\"/> using <see cref=\"T:LibreHardwareMonitor.Hardware.SensorEventHandler\"/>.\n            </summary>\n            <param name=\"sensor\">Sensor class instance that is derived from the <see cref=\"T:LibreHardwareMonitor.Hardware.ISensor\"/> interface.</param>\n        </member>\n        <member name=\"M:LibreHardwareMonitor.Hardware.SensorVisitor.VisitParameter(LibreHardwareMonitor.Hardware.IParameter)\">\n            <summary>\n            Goes through all the components of the specified <see cref=\"T:LibreHardwareMonitor.Hardware.IParameter\"/>.\n            <para>\n            <see cref=\"T:System.NotImplementedException\"/>\n            </para>\n            </summary>\n            <param name=\"parameter\">Parameter class instance that is derived from the <see cref=\"T:LibreHardwareMonitor.Hardware.IParameter\"/> interface.</param>\n        </member>\n        <member name=\"T:LibreHardwareMonitor.Hardware.ChassisSecurityStatus\">\n            <summary>\n            Chassis security status based on <see href=\"https://www.dmtf.org/dsp/DSP0134\">DMTF SMBIOS Reference Specification v.3.3.0, Chapter 7.4.3</see>.\n            </summary>\n        </member>\n        <member name=\"T:LibreHardwareMonitor.Hardware.ChassisStates\">\n            <summary>\n            Chassis state based on <see href=\"https://www.dmtf.org/dsp/DSP0134\">DMTF SMBIOS Reference Specification v.3.3.0, Chapter 7.4.2</see>.\n            </summary>\n        </member>\n        <member name=\"T:LibreHardwareMonitor.Hardware.ChassisType\">\n            <summary>\n            Chassis type based on <see href=\"https://www.dmtf.org/dsp/DSP0134\">DMTF SMBIOS Reference Specification v.3.3.0, Chapter 7.4.1</see>.\n            </summary>\n        </member>\n        <member name=\"T:LibreHardwareMonitor.Hardware.ProcessorFamily\">\n            <summary>\n            Processor family based on <see href=\"https://www.dmtf.org/dsp/DSP0134\">DMTF SMBIOS Reference Specification v.3.3.0, Chapter 7.5.2</see>.\n            </summary>\n        </member>\n        <member name=\"T:LibreHardwareMonitor.Hardware.ProcessorType\">\n            <summary>\n            Processor type based on <see href=\"https://www.dmtf.org/dsp/DSP0134\">DMTF SMBIOS Reference Specification v.3.3.0, Chapter 7.5.1</see>.\n            </summary>\n        </member>\n        <member name=\"T:LibreHardwareMonitor.Hardware.ProcessorSocket\">\n            <summary>\n            Processor socket based on <see href=\"https://www.dmtf.org/dsp/DSP0134\">DMTF SMBIOS Reference Specification v.3.3.0, Chapter 7.5.5</see>.\n            </summary>\n        </member>\n        <member name=\"T:LibreHardwareMonitor.Hardware.SystemWakeUp\">\n            <summary>\n            System wake-up type based on <see href=\"https://www.dmtf.org/dsp/DSP0134\">DMTF SMBIOS Reference Specification v.3.3.0, Chapter 7.2.2</see>.\n            </summary>\n        </member>\n        <member name=\"T:LibreHardwareMonitor.Hardware.CacheAssociativity\">\n            <summary>\n            Cache associativity based on <see href=\"https://www.dmtf.org/dsp/DSP0134\">DMTF SMBIOS Reference Specification v.3.3.0, Chapter 7.8.5</see>.\n            </summary>\n        </member>\n        <member name=\"T:LibreHardwareMonitor.Hardware.CacheDesignation\">\n            <summary>\n            Processor cache level.\n            </summary>\n        </member>\n        <member name=\"M:LibreHardwareMonitor.Hardware.InformationBase.#ctor(System.Byte[],System.Collections.Generic.IList{System.String})\">\n            <summary>\n            Initializes a new instance of the <see cref=\"T:LibreHardwareMonitor.Hardware.InformationBase\" /> class.\n            </summary>\n            <param name=\"data\">The data.</param>\n            <param name=\"strings\">The strings.</param>\n        </member>\n        <member name=\"M:LibreHardwareMonitor.Hardware.InformationBase.GetByte(System.Int32)\">\n            <summary>\n            Gets the byte.\n            </summary>\n            <param name=\"offset\">The offset.</param>\n            <returns><see cref=\"T:System.Int32\" />.</returns>\n        </member>\n        <member name=\"M:LibreHardwareMonitor.Hardware.InformationBase.GetWord(System.Int32)\">\n            <summary>\n            Gets the word.\n            </summary>\n            <param name=\"offset\">The offset.</param>\n            <returns><see cref=\"T:System.Int32\" />.</returns>\n        </member>\n        <member name=\"M:LibreHardwareMonitor.Hardware.InformationBase.GetString(System.Int32)\">\n            <summary>\n            Gets the string.\n            </summary>\n            <param name=\"offset\">The offset.</param>\n            <returns><see cref=\"T:System.String\" />.</returns>\n        </member>\n        <member name=\"T:LibreHardwareMonitor.Hardware.BiosInformation\">\n            <summary>\n            Motherboard BIOS information obtained from the SMBIOS table.\n            </summary>\n        </member>\n        <member name=\"P:LibreHardwareMonitor.Hardware.BiosInformation.Date\">\n            <summary>\n            Gets the BIOS release date.\n            </summary>\n        </member>\n        <member name=\"P:LibreHardwareMonitor.Hardware.BiosInformation.Size\">\n            <summary>\n            Gets the size of the physical device containing the BIOS.\n            </summary>\n        </member>\n        <member name=\"P:LibreHardwareMonitor.Hardware.BiosInformation.Vendor\">\n            <summary>\n            Gets the string number of the BIOS Vendor’s Name.\n            </summary>\n        </member>\n        <member name=\"P:LibreHardwareMonitor.Hardware.BiosInformation.Version\">\n            <summary>\n            Gets the string number of the BIOS Version. This value is a free-form string that may contain Core and OEM version information.\n            </summary>\n        </member>\n        <member name=\"M:LibreHardwareMonitor.Hardware.BiosInformation.GetSize\">\n            <summary>\n            Gets the size.\n            </summary>\n            <returns><see cref=\"T:System.Nullable`1\" />.</returns>\n        </member>\n        <member name=\"M:LibreHardwareMonitor.Hardware.BiosInformation.GetDate(System.String)\">\n            <summary>\n            Gets the date.\n            </summary>\n            <param name=\"date\">The bios date.</param>\n            <returns><see cref=\"T:System.Nullable`1\" />.</returns>\n        </member>\n        <member name=\"T:LibreHardwareMonitor.Hardware.SystemInformation\">\n            <summary>\n            System information obtained from the SMBIOS table.\n            </summary>\n        </member>\n        <member name=\"P:LibreHardwareMonitor.Hardware.SystemInformation.Family\">\n            <summary>\n            Gets the family associated with system.\n            <para>\n            This text string identifies the family to which a particular computer belongs. A family refers to a set of computers that are similar but not identical from a hardware or software point of\n            view. Typically, a family is composed of different computer models, which have different configurations and pricing points. Computers in the same family often have similar branding and cosmetic\n            features.\n            </para>\n            </summary>\n        </member>\n        <member name=\"P:LibreHardwareMonitor.Hardware.SystemInformation.ManufacturerName\">\n            <summary>\n            Gets the manufacturer name associated with system.\n            </summary>\n        </member>\n        <member name=\"P:LibreHardwareMonitor.Hardware.SystemInformation.ProductName\">\n            <summary>\n            Gets the product name associated with system.\n            </summary>\n        </member>\n        <member name=\"P:LibreHardwareMonitor.Hardware.SystemInformation.SerialNumber\">\n            <summary>\n            Gets the serial number string associated with system.\n            </summary>\n        </member>\n        <member name=\"P:LibreHardwareMonitor.Hardware.SystemInformation.Version\">\n            <summary>\n            Gets the version string associated with system.\n            </summary>\n        </member>\n        <member name=\"P:LibreHardwareMonitor.Hardware.SystemInformation.WakeUp\">\n            <summary>\n            Gets <inheritdoc cref=\"T:LibreHardwareMonitor.Hardware.SystemWakeUp\" />\n            </summary>\n        </member>\n        <member name=\"T:LibreHardwareMonitor.Hardware.ChassisInformation\">\n            <summary>\n            Chassis information obtained from the SMBIOS table.\n            </summary>\n        </member>\n        <member name=\"P:LibreHardwareMonitor.Hardware.ChassisInformation.AssetTag\">\n            <summary>\n            Gets the asset tag associated with the enclosure or chassis.\n            </summary>\n        </member>\n        <member name=\"P:LibreHardwareMonitor.Hardware.ChassisInformation.BootUpState\">\n            <summary>\n            Gets <inheritdoc cref=\"T:LibreHardwareMonitor.Hardware.ChassisStates\" />\n            </summary>\n        </member>\n        <member name=\"P:LibreHardwareMonitor.Hardware.ChassisInformation.ChassisType\">\n            <summary>\n            Gets <inheritdoc cref=\"T:LibreHardwareMonitor.Hardware.ChassisType\" />\n            </summary>\n        </member>\n        <member name=\"P:LibreHardwareMonitor.Hardware.ChassisInformation.LockDetected\">\n            <summary>\n            Gets or sets the chassis lock.\n            </summary>\n            <returns>Chassis lock is present if <see langword=\"true\" />. Otherwise, either a lock is not present or it is unknown if the enclosure has a lock.</returns>\n        </member>\n        <member name=\"P:LibreHardwareMonitor.Hardware.ChassisInformation.ManufacturerName\">\n            <summary>\n            Gets the string describing the chassis or enclosure manufacturer name.\n            </summary>\n        </member>\n        <member name=\"P:LibreHardwareMonitor.Hardware.ChassisInformation.PowerCords\">\n            <summary>\n            Gets the number of power cords associated with the enclosure or chassis.\n            </summary>\n        </member>\n        <member name=\"P:LibreHardwareMonitor.Hardware.ChassisInformation.PowerSupplyState\">\n            <summary>\n            Gets the state of the enclosure’s power supply (or supplies) when last booted.\n            </summary>\n        </member>\n        <member name=\"P:LibreHardwareMonitor.Hardware.ChassisInformation.RackHeight\">\n            <summary>\n            Gets the height of the enclosure, in 'U's. A U is a standard unit of measure for the height of a rack or rack-mountable component and is equal to 1.75 inches or 4.445 cm. A value of <c>0</c>\n            indicates that the enclosure height is unspecified.\n            </summary>\n        </member>\n        <member name=\"P:LibreHardwareMonitor.Hardware.ChassisInformation.SecurityStatus\">\n            <summary>\n            Gets the physical security status of the enclosure when last booted.\n            </summary>\n        </member>\n        <member name=\"P:LibreHardwareMonitor.Hardware.ChassisInformation.SerialNumber\">\n            <summary>\n            Gets the string describing the chassis or enclosure serial number.\n            </summary>\n        </member>\n        <member name=\"P:LibreHardwareMonitor.Hardware.ChassisInformation.SKU\">\n            <summary>\n            Gets the string describing the chassis or enclosure SKU number.\n            </summary>\n        </member>\n        <member name=\"P:LibreHardwareMonitor.Hardware.ChassisInformation.ThermalState\">\n            <summary>\n            Gets the thermal state of the enclosure when last booted.\n            </summary>\n        </member>\n        <member name=\"P:LibreHardwareMonitor.Hardware.ChassisInformation.Version\">\n            <summary>\n            Gets the number of null-terminated string representing the chassis or enclosure version.\n            </summary>\n        </member>\n        <member name=\"T:LibreHardwareMonitor.Hardware.BaseBoardInformation\">\n            <summary>\n            Motherboard information obtained from the SMBIOS table.\n            </summary>\n        </member>\n        <member name=\"P:LibreHardwareMonitor.Hardware.BaseBoardInformation.ManufacturerName\">\n            <summary>\n            Gets the value that represents the manufacturer's name.\n            </summary>\n        </member>\n        <member name=\"P:LibreHardwareMonitor.Hardware.BaseBoardInformation.ProductName\">\n            <summary>\n            Gets the value that represents the motherboard's name.\n            </summary>\n        </member>\n        <member name=\"P:LibreHardwareMonitor.Hardware.BaseBoardInformation.SerialNumber\">\n            <summary>\n            Gets the value that represents the motherboard's serial number.\n            </summary>\n        </member>\n        <member name=\"P:LibreHardwareMonitor.Hardware.BaseBoardInformation.Version\">\n            <summary>\n            Gets the value that represents the motherboard's revision number.\n            </summary>\n        </member>\n        <member name=\"T:LibreHardwareMonitor.Hardware.ProcessorInformation\">\n            <summary>\n            Processor information obtained from the SMBIOS table.\n            </summary>\n        </member>\n        <member name=\"P:LibreHardwareMonitor.Hardware.ProcessorInformation.CoreCount\">\n            <summary>\n            Gets the value that represents the number of cores per processor socket.\n            </summary>\n        </member>\n        <member name=\"P:LibreHardwareMonitor.Hardware.ProcessorInformation.CoreEnabled\">\n            <summary>\n            Gets the value that represents the number of enabled cores per processor socket.\n            </summary>\n        </member>\n        <member name=\"P:LibreHardwareMonitor.Hardware.ProcessorInformation.CurrentSpeed\">\n            <summary>\n            Gets the value that represents the current processor speed (in MHz).\n            </summary>\n        </member>\n        <member name=\"P:LibreHardwareMonitor.Hardware.ProcessorInformation.ExternalClock\">\n            <summary>\n            Gets the external Clock Frequency, in MHz. If the value is unknown, the field is set to 0.\n            </summary>\n        </member>\n        <member name=\"P:LibreHardwareMonitor.Hardware.ProcessorInformation.Family\">\n            <summary>\n            Gets <inheritdoc cref=\"T:LibreHardwareMonitor.Hardware.ProcessorFamily\" />\n            </summary>\n        </member>\n        <member name=\"P:LibreHardwareMonitor.Hardware.ProcessorInformation.ManufacturerName\">\n            <summary>\n            Gets the string number of Processor Manufacturer.\n            </summary>\n        </member>\n        <member name=\"P:LibreHardwareMonitor.Hardware.ProcessorInformation.MaxSpeed\">\n            <summary>\n            Gets the value that represents the maximum processor speed (in MHz) supported by the system for this processor socket.\n            </summary>\n        </member>\n        <member name=\"P:LibreHardwareMonitor.Hardware.ProcessorInformation.ProcessorType\">\n            <summary>\n            Gets <inheritdoc cref=\"T:LibreHardwareMonitor.Hardware.ProcessorType\" />\n            </summary>\n        </member>\n        <member name=\"P:LibreHardwareMonitor.Hardware.ProcessorInformation.Serial\">\n            <summary>\n            Gets the value that represents the string number for the serial number of this processor.\n            <para>This value is set by the manufacturer and normally not changeable.</para>\n            </summary>\n        </member>\n        <member name=\"P:LibreHardwareMonitor.Hardware.ProcessorInformation.Socket\">\n            <summary>\n            Gets <inheritdoc cref=\"T:LibreHardwareMonitor.Hardware.ProcessorSocket\" />\n            </summary>\n        </member>\n        <member name=\"P:LibreHardwareMonitor.Hardware.ProcessorInformation.SocketDesignation\">\n            <summary>\n            Gets the string number for Reference Designation.\n            </summary>\n        </member>\n        <member name=\"P:LibreHardwareMonitor.Hardware.ProcessorInformation.ThreadCount\">\n            <summary>\n            Gets the value that represents the number of threads per processor socket.\n            </summary>\n        </member>\n        <member name=\"P:LibreHardwareMonitor.Hardware.ProcessorInformation.Version\">\n            <summary>\n            Gets the value that represents the string number describing the Processor.\n            </summary>\n        </member>\n        <member name=\"T:LibreHardwareMonitor.Hardware.ProcessorCache\">\n            <summary>\n            Processor cache information obtained from the SMBIOS table.\n            </summary>\n        </member>\n        <member name=\"P:LibreHardwareMonitor.Hardware.ProcessorCache.Associativity\">\n            <summary>\n            Gets <inheritdoc cref=\"T:LibreHardwareMonitor.Hardware.CacheAssociativity\" />\n            </summary>\n        </member>\n        <member name=\"P:LibreHardwareMonitor.Hardware.ProcessorCache.Designation\">\n            <summary>\n            Gets <inheritdoc cref=\"T:LibreHardwareMonitor.Hardware.CacheDesignation\" />\n            </summary>\n        </member>\n        <member name=\"P:LibreHardwareMonitor.Hardware.ProcessorCache.Size\">\n            <summary>\n            Gets the value that represents the installed cache size.\n            </summary>\n        </member>\n        <member name=\"M:LibreHardwareMonitor.Hardware.ProcessorCache.GetCacheDesignation\">\n            <summary>\n            Gets the cache designation.\n            </summary>\n            <returns><see cref=\"T:LibreHardwareMonitor.Hardware.CacheDesignation\" />.</returns>\n        </member>\n        <member name=\"T:LibreHardwareMonitor.Hardware.MemoryDevice\">\n            <summary>\n            Memory information obtained from the SMBIOS table.\n            </summary>\n        </member>\n        <member name=\"P:LibreHardwareMonitor.Hardware.MemoryDevice.BankLocator\">\n            <summary>\n            Gets the string number of the string that identifies the physically labeled bank where the memory device is located.\n            </summary>\n        </member>\n        <member name=\"P:LibreHardwareMonitor.Hardware.MemoryDevice.DeviceLocator\">\n            <summary>\n            Gets the string number of the string that identifies the physically-labeled socket or board position where the memory device is located.\n            </summary>\n        </member>\n        <member name=\"P:LibreHardwareMonitor.Hardware.MemoryDevice.ManufacturerName\">\n            <summary>\n            Gets the string number for the manufacturer of this memory device.\n            </summary>\n        </member>\n        <member name=\"P:LibreHardwareMonitor.Hardware.MemoryDevice.PartNumber\">\n            <summary>\n            Gets the string number for the part number of this memory device.\n            </summary>\n        </member>\n        <member name=\"P:LibreHardwareMonitor.Hardware.MemoryDevice.SerialNumber\">\n            <summary>\n            Gets the string number for the serial number of this memory device.\n            </summary>\n        </member>\n        <member name=\"P:LibreHardwareMonitor.Hardware.MemoryDevice.Size\">\n            <summary>\n            Gets the size of the memory device. If the value is 0, no memory device is installed in the socket.\n            </summary>\n        </member>\n        <member name=\"P:LibreHardwareMonitor.Hardware.MemoryDevice.Speed\">\n            <summary>\n            Gets the the value that identifies the maximum capable speed of the device, in mega transfers per second (MT/s).\n            </summary>\n        </member>\n        <member name=\"T:LibreHardwareMonitor.Hardware.SMBios\">\n            <summary>\n            Reads and processes information encoded in an SMBIOS table.\n            </summary>\n        </member>\n        <member name=\"P:LibreHardwareMonitor.Hardware.SMBios.Bios\">\n            <summary>\n            Gets <inheritdoc cref=\"T:LibreHardwareMonitor.Hardware.BiosInformation\" />\n            </summary>\n        </member>\n        <member name=\"P:LibreHardwareMonitor.Hardware.SMBios.Board\">\n            <summary>\n            Gets <inheritdoc cref=\"T:LibreHardwareMonitor.Hardware.BaseBoardInformation\" />\n            </summary>\n        </member>\n        <member name=\"P:LibreHardwareMonitor.Hardware.SMBios.Chassis\">\n            <summary>\n            Gets <inheritdoc cref=\"T:LibreHardwareMonitor.Hardware.ChassisInformation\" />\n            </summary>\n        </member>\n        <member name=\"P:LibreHardwareMonitor.Hardware.SMBios.MemoryDevices\">\n            <summary>\n            Gets <inheritdoc cref=\"T:LibreHardwareMonitor.Hardware.MemoryDevice\" />\n            </summary>\n        </member>\n        <member name=\"P:LibreHardwareMonitor.Hardware.SMBios.Processor\">\n            <summary>\n            Gets <inheritdoc cref=\"T:LibreHardwareMonitor.Hardware.ProcessorInformation\" />\n            </summary>\n        </member>\n        <member name=\"P:LibreHardwareMonitor.Hardware.SMBios.ProcessorCaches\">\n            <summary>\n            Gets <inheritdoc cref=\"T:LibreHardwareMonitor.Hardware.ProcessorCache\" />\n            </summary>\n        </member>\n        <member name=\"P:LibreHardwareMonitor.Hardware.SMBios.System\">\n            <summary>\n            Gets <inheritdoc cref=\"T:LibreHardwareMonitor.Hardware.SystemInformation\" />\n            </summary>\n        </member>\n        <member name=\"M:LibreHardwareMonitor.Hardware.SMBios.GetReport\">\n            <summary>\n            Report containing most of the information that could be read from the SMBIOS table.\n            </summary>\n            <returns>A formatted text string with computer information and the entire SMBIOS table.</returns>\n        </member>\n        <member name=\"T:LibreHardwareMonitor.Hardware.Storage.AbstractStorage.PerformanceValue\">\n            <summary>\n            Helper to calculate the disk performance with base timestamps\n            https://docs.microsoft.com/en-us/windows/win32/cimwin32prov/win32-perfrawdata\n            </summary>\n        </member>\n        <member name=\"P:LibreHardwareMonitor.Hardware.Storage.AtaStorage.Smart\">\n            <summary>\n            Gets the SMART data.\n            </summary>\n        </member>\n        <member name=\"P:LibreHardwareMonitor.Hardware.Storage.AtaStorage.SmartAttributes\">\n            <summary>\n            Gets the SMART attributes.\n            </summary>\n        </member>\n        <member name=\"P:LibreHardwareMonitor.Hardware.Storage.NVMeGeneric.Smart\">\n            <summary>\n            Gets the SMART data.\n            </summary>\n        </member>\n        <member name=\"M:LibreHardwareMonitor.Hardware.Storage.SmartAttribute.#ctor(System.Byte,System.String)\">\n            <summary>\n            Initializes a new instance of the <see cref=\"T:LibreHardwareMonitor.Hardware.Storage.SmartAttribute\" /> class.\n            </summary>\n            <param name=\"id\">The SMART id of the attribute.</param>\n            <param name=\"name\">The name of the attribute.</param>\n        </member>\n        <member name=\"M:LibreHardwareMonitor.Hardware.Storage.SmartAttribute.#ctor(System.Byte,System.String,LibreHardwareMonitor.Hardware.Storage.SmartAttribute.RawValueConversion)\">\n            <summary>\n            Initializes a new instance of the <see cref=\"T:LibreHardwareMonitor.Hardware.Storage.SmartAttribute\" /> class.\n            </summary>\n            <param name=\"id\">The SMART id of the attribute.</param>\n            <param name=\"name\">The name of the attribute.</param>\n            <param name=\"rawValueConversion\">\n            A delegate for converting the raw byte\n            array into a value (or null to use the attribute value).\n            </param>\n        </member>\n        <member name=\"M:LibreHardwareMonitor.Hardware.Storage.SmartAttribute.#ctor(System.Byte,System.String,LibreHardwareMonitor.Hardware.Storage.SmartAttribute.RawValueConversion,System.Nullable{LibreHardwareMonitor.Hardware.SensorType},System.Int32,System.String,System.Boolean,LibreHardwareMonitor.Hardware.ParameterDescription[])\">\n            <summary>\n            Initializes a new instance of the <see cref=\"T:LibreHardwareMonitor.Hardware.Storage.SmartAttribute\" /> class.\n            </summary>\n            <param name=\"id\">The SMART id of the attribute.</param>\n            <param name=\"name\">The name of the attribute.</param>\n            <param name=\"rawValueConversion\">\n            A delegate for converting the raw byte\n            array into a value (or null to use the attribute value).\n            </param>\n            <param name=\"sensorType\">\n            Type of the sensor or null if no sensor is to\n            be created.\n            </param>\n            <param name=\"sensorChannel\">\n            If there exists more than one attribute with\n            the same sensor channel and type, then a sensor is created only for the\n            first attribute.\n            </param>\n            <param name=\"sensorName\">\n            The name to be used for the sensor, or null if\n            no sensor is created.\n            </param>\n            <param name=\"defaultHiddenSensor\">True to hide the sensor initially.</param>\n            <param name=\"parameterDescriptions\">\n            Description for the parameters of the sensor\n            (or null).\n            </param>\n        </member>\n        <member name=\"P:LibreHardwareMonitor.Hardware.Storage.SmartAttribute.Id\">\n            <summary>\n            Gets the SMART identifier.\n            </summary>\n        </member>\n        <member name=\"T:LibreHardwareMonitor.Hardware.Storage.SmartNames\">\n            <summary>\n            Localization class for SMART attribute names.\n            </summary>\n        </member>\n        <member name=\"M:LibreHardwareMonitor.Hardware.Storage.SsdMicron.UpdateAdditionalSensors(LibreHardwareMonitor.Interop.Kernel32.SMART_ATTRIBUTE[])\">\n            <inheritdoc />\n        </member>\n        <member name=\"M:LibreHardwareMonitor.Hardware.Storage.WindowsSmart.ReadSmartHealth\">\n            <summary>\n            Reads Smart health status of the drive\n            </summary>\n            <returns>True, if drive is healthy; False, if unhealthy; Null, if it cannot be read</returns>\n        </member>\n        <member name=\"M:LibreHardwareMonitor.Hardware.ThreadAffinity.#cctor\">\n            <summary>\n            Initializes static members of the <see cref=\"T:LibreHardwareMonitor.Hardware.ThreadAffinity\" /> class.\n            </summary>\n        </member>\n        <member name=\"P:LibreHardwareMonitor.Hardware.ThreadAffinity.ProcessorGroupCount\">\n            <summary>\n            Gets the processor group count.\n            </summary>\n        </member>\n        <member name=\"M:LibreHardwareMonitor.Hardware.ThreadAffinity.IsValid(LibreHardwareMonitor.Hardware.GroupAffinity)\">\n            <summary>\n            Returns true if the <paramref name=\"affinity\"/> is valid.\n            </summary>\n            <param name=\"affinity\">The affinity.</param>\n            <returns><c>true</c> if the specified affinity is valid; otherwise, <c>false</c>.</returns>\n        </member>\n        <member name=\"M:LibreHardwareMonitor.Hardware.ThreadAffinity.Set(LibreHardwareMonitor.Hardware.GroupAffinity)\">\n            <summary>\n            Sets the processor group affinity for the current thread.\n            </summary>\n            <param name=\"affinity\">The processor group affinity.</param>\n            <returns>The previous processor group affinity.</returns>\n        </member>\n        <member name=\"F:LibreHardwareMonitor.Interop.AtiAdlxx.ADLStatus.ADL_OK_WAIT\">\n            <summary>\n            All OK, but need to wait.\n            </summary>\n        </member>\n        <member name=\"F:LibreHardwareMonitor.Interop.AtiAdlxx.ADLStatus.ADL_OK_RESTART\">\n            <summary>\n            All OK, but need restart.\n            </summary>\n        </member>\n        <member name=\"F:LibreHardwareMonitor.Interop.AtiAdlxx.ADLStatus.ADL_OK_MODE_CHANGE\">\n            <summary>\n            All OK but need mode change.\n            </summary>\n        </member>\n        <member name=\"F:LibreHardwareMonitor.Interop.AtiAdlxx.ADLStatus.ADL_OK_WARNING\">\n            <summary>\n            All OK, but with warning.\n            </summary>\n        </member>\n        <member name=\"F:LibreHardwareMonitor.Interop.AtiAdlxx.ADLStatus.ADL_OK\">\n            <summary>\n            ADL function completed successfully.\n            </summary>\n        </member>\n        <member name=\"F:LibreHardwareMonitor.Interop.AtiAdlxx.ADLStatus.ADL_ERR\">\n            <summary>\n            Generic Error. Most likely one or more of the Escape calls to the driver\n            failed!\n            </summary>\n        </member>\n        <member name=\"F:LibreHardwareMonitor.Interop.AtiAdlxx.ADLStatus.ADL_ERR_NOT_INIT\">\n            <summary>\n            ADL not initialized.\n            </summary>\n        </member>\n        <member name=\"F:LibreHardwareMonitor.Interop.AtiAdlxx.ADLStatus.ADL_ERR_INVALID_PARAM\">\n            <summary>\n            One of the parameter passed is invalid.\n            </summary>\n        </member>\n        <member name=\"F:LibreHardwareMonitor.Interop.AtiAdlxx.ADLStatus.ADL_ERR_INVALID_PARAM_SIZE\">\n            <summary>\n            One of the parameter size is invalid.\n            </summary>\n        </member>\n        <member name=\"F:LibreHardwareMonitor.Interop.AtiAdlxx.ADLStatus.ADL_ERR_INVALID_ADL_IDX\">\n            <summary>\n            Invalid ADL index passed.\n            </summary>\n        </member>\n        <member name=\"F:LibreHardwareMonitor.Interop.AtiAdlxx.ADLStatus.ADL_ERR_INVALID_CONTROLLER_IDX\">\n            <summary>\n            Invalid controller index passed.\n            </summary>\n        </member>\n        <member name=\"F:LibreHardwareMonitor.Interop.AtiAdlxx.ADLStatus.ADL_ERR_INVALID_DIPLAY_IDX\">\n            <summary>\n            Invalid display index passed.\n            </summary>\n        </member>\n        <member name=\"F:LibreHardwareMonitor.Interop.AtiAdlxx.ADLStatus.ADL_ERR_NOT_SUPPORTED\">\n            <summary>\n            Function not supported by the driver.\n            </summary>\n        </member>\n        <member name=\"F:LibreHardwareMonitor.Interop.AtiAdlxx.ADLStatus.ADL_ERR_NULL_POINTER\">\n            <summary>\n            Null Pointer error.\n            </summary>\n        </member>\n        <member name=\"F:LibreHardwareMonitor.Interop.AtiAdlxx.ADLStatus.ADL_ERR_DISABLED_ADAPTER\">\n            <summary>\n            Call can't be made due to disabled adapter.\n            </summary>\n        </member>\n        <member name=\"F:LibreHardwareMonitor.Interop.AtiAdlxx.ADLStatus.ADL_ERR_INVALID_CALLBACK\">\n            <summary>\n            Invalid Callback.\n            </summary>\n        </member>\n        <member name=\"F:LibreHardwareMonitor.Interop.AtiAdlxx.ADLStatus.ADL_ERR_RESOURCE_CONFLICT\">\n            <summary>\n            Display Resource conflict.\n            </summary>\n        </member>\n        <member name=\"F:LibreHardwareMonitor.Interop.AtiAdlxx.ADLStatus.ADL_ERR_SET_INCOMPLETE\">\n            <summary>\n            Failed to update some of the values. Can be returned by set request that\n            include multiple values if not all values were successfully committed.\n            </summary>\n        </member>\n        <member name=\"F:LibreHardwareMonitor.Interop.AtiAdlxx.ADLStatus.ADL_ERR_NO_XDISPLAY\">\n            <summary>\n            There's no Linux XDisplay in Linux Console environment.\n            </summary>\n        </member>\n        <member name=\"F:LibreHardwareMonitor.Interop.Kernel32.NVME_CRITICAL_WARNING.AvailableSpaceLow\">\n            <summary>\n            If set to 1, then the available spare space has fallen below the threshold.\n            </summary>\n        </member>\n        <member name=\"F:LibreHardwareMonitor.Interop.Kernel32.NVME_CRITICAL_WARNING.TemperatureThreshold\">\n            <summary>\n            If set to 1, then a temperature is above an over temperature threshold or below an under temperature threshold.\n            </summary>\n        </member>\n        <member name=\"F:LibreHardwareMonitor.Interop.Kernel32.NVME_CRITICAL_WARNING.ReliabilityDegraded\">\n            <summary>\n            If set to 1, then the device reliability has been degraded due to significant media related errors or any internal error that degrades device reliability.\n            </summary>\n        </member>\n        <member name=\"F:LibreHardwareMonitor.Interop.Kernel32.NVME_CRITICAL_WARNING.ReadOnly\">\n            <summary>\n            If set to 1, then the media has been placed in read only mode\n            </summary>\n        </member>\n        <member name=\"F:LibreHardwareMonitor.Interop.Kernel32.NVME_CRITICAL_WARNING.VolatileMemoryBackupDeviceFailed\">\n            <summary>\n            If set to 1, then the volatile memory backup device has failed. This field is only valid if the controller has a volatile memory backup solution.\n            </summary>\n        </member>\n        <member name=\"M:LibreHardwareMonitor.Interop.Kernel32.CreateStruct``1\">\n            <summary>\n            Create a instance from a struct with zero initialized memory arrays\n            no need to init every inner array with the correct sizes\n            </summary>\n            <typeparam name=\"T\">type of struct that is needed</typeparam>\n            <returns></returns>\n        </member>\n        <member name=\"F:LibreHardwareMonitor.Interop.Kernel32.ATA_COMMAND.ATA_SMART\">\n            <summary>\n            SMART data requested.\n            </summary>\n        </member>\n        <member name=\"F:LibreHardwareMonitor.Interop.Kernel32.ATA_COMMAND.ATA_IDENTIFY_DEVICE\">\n            <summary>\n            Identify data is requested.\n            </summary>\n        </member>\n        <member name=\"F:LibreHardwareMonitor.Interop.Kernel32.SMART_FEATURES.SMART_READ_DATA\">\n            <summary>\n            Read SMART data.\n            </summary>\n        </member>\n        <member name=\"F:LibreHardwareMonitor.Interop.Kernel32.SMART_FEATURES.READ_THRESHOLDS\">\n            <summary>\n            Read SMART thresholds.\n            obsolete\n            </summary>\n        </member>\n        <member name=\"F:LibreHardwareMonitor.Interop.Kernel32.SMART_FEATURES.ENABLE_DISABLE_AUTOSAVE\">\n            <summary>\n            Autosave SMART data.\n            </summary>\n        </member>\n        <member name=\"F:LibreHardwareMonitor.Interop.Kernel32.SMART_FEATURES.SAVE_ATTRIBUTE_VALUES\">\n            <summary>\n            Save SMART attributes.\n            </summary>\n        </member>\n        <member name=\"F:LibreHardwareMonitor.Interop.Kernel32.SMART_FEATURES.EXECUTE_OFFLINE_DIAGS\">\n            <summary>\n            Set SMART to offline immediately.\n            </summary>\n        </member>\n        <member name=\"F:LibreHardwareMonitor.Interop.Kernel32.SMART_FEATURES.SMART_READ_LOG\">\n            <summary>\n            Read SMART log.\n            </summary>\n        </member>\n        <member name=\"F:LibreHardwareMonitor.Interop.Kernel32.SMART_FEATURES.SMART_WRITE_LOG\">\n            <summary>\n            Write SMART log.\n            </summary>\n        </member>\n        <member name=\"F:LibreHardwareMonitor.Interop.Kernel32.SMART_FEATURES.WRITE_THRESHOLDS\">\n            <summary>\n            Write SMART thresholds.\n            obsolete\n            </summary>\n        </member>\n        <member name=\"F:LibreHardwareMonitor.Interop.Kernel32.SMART_FEATURES.ENABLE_SMART\">\n            <summary>\n            Enable SMART.\n            </summary>\n        </member>\n        <member name=\"F:LibreHardwareMonitor.Interop.Kernel32.SMART_FEATURES.DISABLE_SMART\">\n            <summary>\n            Disable SMART.\n            </summary>\n        </member>\n        <member name=\"F:LibreHardwareMonitor.Interop.Kernel32.SMART_FEATURES.RETURN_SMART_STATUS\">\n            <summary>\n            Get SMART status.\n            </summary>\n        </member>\n        <member name=\"F:LibreHardwareMonitor.Interop.Kernel32.SMART_FEATURES.ENABLE_DISABLE_AUTO_OFFLINE\">\n            <summary>\n            Set SMART to offline automatically.\n            </summary>\n        </member>\n        <member name=\"F:LibreHardwareMonitor.Interop.Kernel32.NVME_POWER_STATE_DESC.MP\">\n            <summary>\n            bit 0:15 Maximum  Power (MP) in centiwatts\n            </summary>\n        </member>\n        <member name=\"F:LibreHardwareMonitor.Interop.Kernel32.NVME_POWER_STATE_DESC.Reserved0\">\n            <summary>\n            bit 16:23\n            </summary>\n        </member>\n        <member name=\"F:LibreHardwareMonitor.Interop.Kernel32.NVME_POWER_STATE_DESC.MPS_NOPS\">\n            <summary>\n            bit 24 Max Power Scale (MPS), bit 25 Non-Operational State (NOPS)\n            </summary>\n        </member>\n        <member name=\"F:LibreHardwareMonitor.Interop.Kernel32.NVME_POWER_STATE_DESC.ENLAT\">\n            <summary>\n            bit 32:63 Entry Latency (ENLAT) in microseconds\n            </summary>\n        </member>\n        <member name=\"F:LibreHardwareMonitor.Interop.Kernel32.NVME_POWER_STATE_DESC.EXLAT\">\n            <summary>\n            bit 64:95 Exit Latency (EXLAT) in microseconds\n            </summary>\n        </member>\n        <member name=\"F:LibreHardwareMonitor.Interop.Kernel32.NVME_POWER_STATE_DESC.RRT\">\n            <summary>\n            bit 96:100 Relative Read Throughput (RRT)\n            </summary>\n        </member>\n        <member name=\"F:LibreHardwareMonitor.Interop.Kernel32.NVME_POWER_STATE_DESC.RRL\">\n            <summary>\n            bit 104:108 Relative Read Latency (RRL)\n            </summary>\n        </member>\n        <member name=\"F:LibreHardwareMonitor.Interop.Kernel32.NVME_POWER_STATE_DESC.RWT\">\n            <summary>\n            bit 112:116 Relative Write Throughput (RWT)\n            </summary>\n        </member>\n        <member name=\"F:LibreHardwareMonitor.Interop.Kernel32.NVME_POWER_STATE_DESC.RWL\">\n            <summary>\n            bit 120:124 Relative Write Latency (RWL)\n            </summary>\n        </member>\n        <member name=\"F:LibreHardwareMonitor.Interop.Kernel32.NVME_POWER_STATE_DESC.IDLP\">\n            <summary>\n            bit 128:143 Idle Power (IDLP)\n            </summary>\n        </member>\n        <member name=\"F:LibreHardwareMonitor.Interop.Kernel32.NVME_POWER_STATE_DESC.IPS\">\n            <summary>\n            bit 150:151 Idle Power Scale (IPS)\n            </summary>\n        </member>\n        <member name=\"F:LibreHardwareMonitor.Interop.Kernel32.NVME_POWER_STATE_DESC.Reserved7\">\n            <summary>\n            bit 152:159\n            </summary>\n        </member>\n        <member name=\"F:LibreHardwareMonitor.Interop.Kernel32.NVME_POWER_STATE_DESC.ACTP\">\n            <summary>\n            bit 160:175 Active Power (ACTP)\n            </summary>\n        </member>\n        <member name=\"F:LibreHardwareMonitor.Interop.Kernel32.NVME_POWER_STATE_DESC.APW_APS\">\n            <summary>\n            bit 176:178 Active Power Workload (APW), bit 182:183  Active Power Scale (APS)\n            </summary>\n        </member>\n        <member name=\"F:LibreHardwareMonitor.Interop.Kernel32.NVME_POWER_STATE_DESC.Reserved9\">\n            <summary>\n            bit 184:255.\n            </summary>\n        </member>\n        <member name=\"F:LibreHardwareMonitor.Interop.Kernel32.NVME_IDENTIFY_CONTROLLER_DATA.VID\">\n            <summary>\n            byte 0:1 M - PCI Vendor ID (VID)\n            </summary>\n        </member>\n        <member name=\"F:LibreHardwareMonitor.Interop.Kernel32.NVME_IDENTIFY_CONTROLLER_DATA.SSVID\">\n            <summary>\n            byte 2:3 M - PCI Subsystem Vendor ID (SSVID)\n            </summary>\n        </member>\n        <member name=\"F:LibreHardwareMonitor.Interop.Kernel32.NVME_IDENTIFY_CONTROLLER_DATA.SN\">\n            <summary>\n            byte 4: 23 M - Serial Number (SN)\n            </summary>\n        </member>\n        <member name=\"F:LibreHardwareMonitor.Interop.Kernel32.NVME_IDENTIFY_CONTROLLER_DATA.MN\">\n            <summary>\n            byte 24:63 M - Model Number (MN)\n            </summary>\n        </member>\n        <member name=\"F:LibreHardwareMonitor.Interop.Kernel32.NVME_IDENTIFY_CONTROLLER_DATA.FR\">\n            <summary>\n            byte 64:71 M - Firmware Revision (FR)\n            </summary>\n        </member>\n        <member name=\"F:LibreHardwareMonitor.Interop.Kernel32.NVME_IDENTIFY_CONTROLLER_DATA.RAB\">\n            <summary>\n            byte 72 M - Recommended Arbitration Burst (RAB)\n            </summary>\n        </member>\n        <member name=\"F:LibreHardwareMonitor.Interop.Kernel32.NVME_IDENTIFY_CONTROLLER_DATA.IEEE\">\n            <summary>\n            byte 73:75 M - IEEE OUI Identifier (IEEE). Controller Vendor code.\n            </summary>\n        </member>\n        <member name=\"F:LibreHardwareMonitor.Interop.Kernel32.NVME_IDENTIFY_CONTROLLER_DATA.CMIC\">\n            <summary>\n            byte 76 O - Controller Multi-Path I/O and Namespace Sharing Capabilities (CMIC)\n            </summary>\n        </member>\n        <member name=\"F:LibreHardwareMonitor.Interop.Kernel32.NVME_IDENTIFY_CONTROLLER_DATA.MDTS\">\n            <summary>\n            byte 77 M - Maximum Data Transfer Size (MDTS)\n            </summary>\n        </member>\n        <member name=\"F:LibreHardwareMonitor.Interop.Kernel32.NVME_IDENTIFY_CONTROLLER_DATA.CNTLID\">\n            <summary>\n            byte 78:79 M - Controller ID (CNTLID)\n            </summary>\n        </member>\n        <member name=\"F:LibreHardwareMonitor.Interop.Kernel32.NVME_IDENTIFY_CONTROLLER_DATA.VER\">\n            <summary>\n            byte 80:83 M - Version (VER)\n            </summary>\n        </member>\n        <member name=\"F:LibreHardwareMonitor.Interop.Kernel32.NVME_IDENTIFY_CONTROLLER_DATA.RTD3R\">\n            <summary>\n            byte 84:87 M - RTD3 Resume Latency (RTD3R)\n            </summary>\n        </member>\n        <member name=\"F:LibreHardwareMonitor.Interop.Kernel32.NVME_IDENTIFY_CONTROLLER_DATA.RTD3E\">\n            <summary>\n            byte 88:91 M - RTD3 Entry Latency (RTD3E)\n            </summary>\n        </member>\n        <member name=\"F:LibreHardwareMonitor.Interop.Kernel32.NVME_IDENTIFY_CONTROLLER_DATA.OAES\">\n            <summary>\n            byte 92:95 M - Optional Asynchronous Events Supported (OAES)\n            </summary>\n        </member>\n        <member name=\"F:LibreHardwareMonitor.Interop.Kernel32.NVME_IDENTIFY_CONTROLLER_DATA.Reserved0\">\n            <summary>\n            byte 96:239.\n            </summary>\n        </member>\n        <member name=\"F:LibreHardwareMonitor.Interop.Kernel32.NVME_IDENTIFY_CONTROLLER_DATA.ReservedForManagement\">\n            <summary>\n            byte 240:255.  Refer to the NVMe Management Interface Specification for definition.\n            </summary>\n        </member>\n        <member name=\"F:LibreHardwareMonitor.Interop.Kernel32.NVME_IDENTIFY_CONTROLLER_DATA.OACS\">\n            <summary>\n            byte 256:257 M - Optional Admin Command Support (OACS)\n            </summary>\n        </member>\n        <member name=\"F:LibreHardwareMonitor.Interop.Kernel32.NVME_IDENTIFY_CONTROLLER_DATA.ACL\">\n            <summary>\n            byte 258 M - Abort Command Limit (ACL)\n            </summary>\n        </member>\n        <member name=\"F:LibreHardwareMonitor.Interop.Kernel32.NVME_IDENTIFY_CONTROLLER_DATA.AERL\">\n            <summary>\n            byte 259 M - Asynchronous Event Request Limit (AERL)\n            </summary>\n        </member>\n        <member name=\"F:LibreHardwareMonitor.Interop.Kernel32.NVME_IDENTIFY_CONTROLLER_DATA.FRMW\">\n            <summary>\n            byte 260 M - Firmware Updates (FRMW)\n            </summary>\n        </member>\n        <member name=\"F:LibreHardwareMonitor.Interop.Kernel32.NVME_IDENTIFY_CONTROLLER_DATA.LPA\">\n            <summary>\n            byte 261 M - Log Page Attributes (LPA)\n            </summary>\n        </member>\n        <member name=\"F:LibreHardwareMonitor.Interop.Kernel32.NVME_IDENTIFY_CONTROLLER_DATA.ELPE\">\n            <summary>\n            byte 262 M - Error Log Page Entries (ELPE)\n            </summary>\n        </member>\n        <member name=\"F:LibreHardwareMonitor.Interop.Kernel32.NVME_IDENTIFY_CONTROLLER_DATA.NPSS\">\n            <summary>\n            byte 263 M - Number of Power States Support (NPSS)\n            </summary>\n        </member>\n        <member name=\"F:LibreHardwareMonitor.Interop.Kernel32.NVME_IDENTIFY_CONTROLLER_DATA.AVSCC\">\n            <summary>\n            byte 264 M - Admin Vendor Specific Command Configuration (AVSCC)\n            </summary>\n        </member>\n        <member name=\"F:LibreHardwareMonitor.Interop.Kernel32.NVME_IDENTIFY_CONTROLLER_DATA.APSTA\">\n            <summary>\n            byte 265 O - Autonomous Power State Transition Attributes (APSTA)\n            </summary>\n        </member>\n        <member name=\"F:LibreHardwareMonitor.Interop.Kernel32.NVME_IDENTIFY_CONTROLLER_DATA.WCTEMP\">\n            <summary>\n            byte 266:267 M - Warning Composite Temperature Threshold (WCTEMP)\n            </summary>\n        </member>\n        <member name=\"F:LibreHardwareMonitor.Interop.Kernel32.NVME_IDENTIFY_CONTROLLER_DATA.CCTEMP\">\n            <summary>\n            byte 268:269 M - Critical Composite Temperature Threshold (CCTEMP)\n            </summary>\n        </member>\n        <member name=\"F:LibreHardwareMonitor.Interop.Kernel32.NVME_IDENTIFY_CONTROLLER_DATA.MTFA\">\n            <summary>\n            byte 270:271 O - Maximum Time for Firmware Activation (MTFA)\n            </summary>\n        </member>\n        <member name=\"F:LibreHardwareMonitor.Interop.Kernel32.NVME_IDENTIFY_CONTROLLER_DATA.HMPRE\">\n            <summary>\n            byte 272:275 O - Host Memory Buffer Preferred Size (HMPRE)\n            </summary>\n        </member>\n        <member name=\"F:LibreHardwareMonitor.Interop.Kernel32.NVME_IDENTIFY_CONTROLLER_DATA.HMMIN\">\n            <summary>\n            byte 276:279 O - Host Memory Buffer Minimum Size (HMMIN)\n            </summary>\n        </member>\n        <member name=\"F:LibreHardwareMonitor.Interop.Kernel32.NVME_IDENTIFY_CONTROLLER_DATA.TNVMCAP\">\n            <summary>\n            byte 280:295 O - Total NVM Capacity (TNVMCAP)\n            </summary>\n        </member>\n        <member name=\"F:LibreHardwareMonitor.Interop.Kernel32.NVME_IDENTIFY_CONTROLLER_DATA.UNVMCAP\">\n            <summary>\n            byte 296:311 O - Unallocated NVM Capacity (UNVMCAP)\n            </summary>\n        </member>\n        <member name=\"F:LibreHardwareMonitor.Interop.Kernel32.NVME_IDENTIFY_CONTROLLER_DATA.RPMBS\">\n            <summary>\n            byte 312:315 O - Replay Protected Memory Block Support (RPMBS)\n            </summary>\n        </member>\n        <member name=\"F:LibreHardwareMonitor.Interop.Kernel32.NVME_IDENTIFY_CONTROLLER_DATA.Reserved1\">\n            <summary>\n            byte 316:511\n            </summary>\n        </member>\n        <member name=\"F:LibreHardwareMonitor.Interop.Kernel32.NVME_IDENTIFY_CONTROLLER_DATA.SQES\">\n            <summary>\n            byte 512 M - Submission Queue Entry Size (SQES)\n            </summary>\n        </member>\n        <member name=\"F:LibreHardwareMonitor.Interop.Kernel32.NVME_IDENTIFY_CONTROLLER_DATA.CQES\">\n            <summary>\n            byte 513 M - Completion Queue Entry Size (CQES)\n            </summary>\n        </member>\n        <member name=\"F:LibreHardwareMonitor.Interop.Kernel32.NVME_IDENTIFY_CONTROLLER_DATA.Reserved2\">\n            <summary>\n            byte 514:515\n            </summary>\n        </member>\n        <member name=\"F:LibreHardwareMonitor.Interop.Kernel32.NVME_IDENTIFY_CONTROLLER_DATA.NN\">\n            <summary>\n            byte 516:519 M - Number of Namespaces (NN)\n            </summary>\n        </member>\n        <member name=\"F:LibreHardwareMonitor.Interop.Kernel32.NVME_IDENTIFY_CONTROLLER_DATA.ONCS\">\n            <summary>\n            byte 520:521 M - Optional NVM Command Support (ONCS)\n            </summary>\n        </member>\n        <member name=\"F:LibreHardwareMonitor.Interop.Kernel32.NVME_IDENTIFY_CONTROLLER_DATA.FUSES\">\n            <summary>\n            byte 522:523 M - Fused Operation Support (FUSES)\n            </summary>\n        </member>\n        <member name=\"F:LibreHardwareMonitor.Interop.Kernel32.NVME_IDENTIFY_CONTROLLER_DATA.FNA\">\n            <summary>\n            byte 524 M - Format NVM Attributes (FNA)\n            </summary>\n        </member>\n        <member name=\"F:LibreHardwareMonitor.Interop.Kernel32.NVME_IDENTIFY_CONTROLLER_DATA.VWC\">\n            <summary>\n            byte 525 M - Volatile Write Cache (VWC)\n            </summary>\n        </member>\n        <member name=\"F:LibreHardwareMonitor.Interop.Kernel32.NVME_IDENTIFY_CONTROLLER_DATA.AWUN\">\n            <summary>\n            byte 526:527 M - Atomic Write Unit Normal (AWUN)\n            </summary>\n        </member>\n        <member name=\"F:LibreHardwareMonitor.Interop.Kernel32.NVME_IDENTIFY_CONTROLLER_DATA.AWUPF\">\n            <summary>\n            byte 528:529 M - Atomic Write Unit Power Fail (AWUPF)\n            </summary>\n        </member>\n        <member name=\"F:LibreHardwareMonitor.Interop.Kernel32.NVME_IDENTIFY_CONTROLLER_DATA.NVSCC\">\n            <summary>\n            byte 530 M - NVM Vendor Specific Command Configuration (NVSCC)\n            </summary>\n        </member>\n        <member name=\"F:LibreHardwareMonitor.Interop.Kernel32.NVME_IDENTIFY_CONTROLLER_DATA.Reserved3\">\n            <summary>\n            byte 531\n            </summary>\n        </member>\n        <!-- Badly formed XML comment ignored for member \"F:LibreHardwareMonitor.Interop.Kernel32.NVME_IDENTIFY_CONTROLLER_DATA.ACWU\" -->\n        <member name=\"F:LibreHardwareMonitor.Interop.Kernel32.NVME_IDENTIFY_CONTROLLER_DATA.Reserved4\">\n            <summary>\n            byte 534:535\n            </summary>\n        </member>\n        <member name=\"F:LibreHardwareMonitor.Interop.Kernel32.NVME_IDENTIFY_CONTROLLER_DATA.SGLS\">\n            <summary>\n            byte 536:539 O - SGL Support (SGLS)\n            </summary>\n        </member>\n        <member name=\"F:LibreHardwareMonitor.Interop.Kernel32.NVME_IDENTIFY_CONTROLLER_DATA.Reserved5\">\n            <summary>\n            byte 540:703\n            </summary>\n        </member>\n        <member name=\"F:LibreHardwareMonitor.Interop.Kernel32.NVME_IDENTIFY_CONTROLLER_DATA.Reserved6\">\n            <summary>\n            byte 704:2047\n            </summary>\n        </member>\n        <member name=\"F:LibreHardwareMonitor.Interop.Kernel32.NVME_IDENTIFY_CONTROLLER_DATA.PDS\">\n            <summary>\n            byte 2048:3071 Power State Descriptors\n            </summary>\n        </member>\n        <member name=\"F:LibreHardwareMonitor.Interop.Kernel32.NVME_IDENTIFY_CONTROLLER_DATA.VS\">\n            <summary>\n            byte 3072:4095 Vendor Specific\n            </summary>\n        </member>\n        <member name=\"F:LibreHardwareMonitor.Interop.Kernel32.NVME_HEALTH_INFO_LOG.CriticalWarning\">\n            <summary>\n            This field indicates critical warnings for the state of the  controller.\n            Each bit corresponds to a critical warning type; multiple bits may be set.\n            </summary>\n        </member>\n        <member name=\"F:LibreHardwareMonitor.Interop.Kernel32.NVME_HEALTH_INFO_LOG.CompositeTemp\">\n            <summary>\n            Composite Temperature:  Contains the temperature of the overall device (controller and NVM included) in units of Kelvin.\n            </summary>\n        </member>\n        <member name=\"F:LibreHardwareMonitor.Interop.Kernel32.NVME_HEALTH_INFO_LOG.AvailableSpare\">\n            <summary>\n            Available Spare:  Contains a normalized percentage (0 to 100%) of the remaining spare capacity available\n            </summary>\n        </member>\n        <member name=\"F:LibreHardwareMonitor.Interop.Kernel32.NVME_HEALTH_INFO_LOG.AvailableSpareThreshold\">\n            <summary>\n            Available Spare Threshold:  When the Available Spare falls below the threshold indicated in this field,\n            an asynchronous event completion may occur. The value is indicated as a normalized percentage (0 to 100%).\n            </summary>\n        </member>\n        <member name=\"F:LibreHardwareMonitor.Interop.Kernel32.NVME_HEALTH_INFO_LOG.PercentageUsed\">\n            <summary>\n            Percentage Used:  Contains a vendor specific estimate of the percentage of NVM subsystem life used based on\n            the actual usage and the manufacturer’s prediction of NVM life. A value of 100 indicates that the estimated endurance of\n            the NVM in the NVM subsystem has been consumed, but may not indicate an NVM subsystem failure. The value is allowed to exceed 100.\n            </summary>\n        </member>\n        <member name=\"F:LibreHardwareMonitor.Interop.Kernel32.NVME_HEALTH_INFO_LOG.DataUnitRead\">\n            <summary>\n            Data Units Read:  Contains the number of 512 byte data units the host has read from the controller;\n            this value does not include metadata. This value is reported in thousands\n            (i.e., a value of 1 corresponds to 1000 units of 512 bytes read) and is rounded up.\n            </summary>\n        </member>\n        <member name=\"F:LibreHardwareMonitor.Interop.Kernel32.NVME_HEALTH_INFO_LOG.DataUnitWritten\">\n            <summary>\n            Data Units Written:  Contains the number of 512 byte data units the host has written to the controller;\n            this value does not include metadata. This value is reported in thousands\n            (i.e., a value of 1 corresponds to 1000 units of 512 bytes written) and is rounded up.\n            </summary>\n        </member>\n        <member name=\"F:LibreHardwareMonitor.Interop.Kernel32.NVME_HEALTH_INFO_LOG.HostReadCommands\">\n            <summary>\n            Host Read Commands:  Contains the number of read commands completed by the controller.\n            For the NVM command set, this is the number of Compare and Read commands.\n            </summary>\n        </member>\n        <member name=\"F:LibreHardwareMonitor.Interop.Kernel32.NVME_HEALTH_INFO_LOG.HostWriteCommands\">\n            <summary>\n            Host Write Commands:  Contains the number of write commands completed by the controller.\n            For the NVM command set, this is the number of Write commands.\n            </summary>\n        </member>\n        <member name=\"F:LibreHardwareMonitor.Interop.Kernel32.NVME_HEALTH_INFO_LOG.ControllerBusyTime\">\n            <summary>\n            Controller Busy Time:  Contains the amount of time the controller is busy with I/O commands.\n            </summary>\n        </member>\n        <member name=\"F:LibreHardwareMonitor.Interop.Kernel32.NVME_HEALTH_INFO_LOG.PowerCycles\">\n            <summary>\n            Power Cycles:  Contains the number of power cycles.\n            </summary>\n        </member>\n        <member name=\"F:LibreHardwareMonitor.Interop.Kernel32.NVME_HEALTH_INFO_LOG.PowerOnHours\">\n            <summary>\n            Power On Hours:  Contains the number of power-on hours.\n            This does not include time that the controller was powered and in a low power state condition.\n            </summary>\n        </member>\n        <member name=\"F:LibreHardwareMonitor.Interop.Kernel32.NVME_HEALTH_INFO_LOG.UnsafeShutdowns\">\n            <summary>\n            Unsafe Shutdowns:  Contains the number of unsafe shutdowns.\n            This count is incremented when a shutdown notification is not received prior to loss of power.\n            </summary>\n        </member>\n        <member name=\"F:LibreHardwareMonitor.Interop.Kernel32.NVME_HEALTH_INFO_LOG.MediaAndDataIntegrityErrors\">\n            <summary>\n            Media Errors:  Contains the number of occurrences where the controller detected an unrecoverable data integrity error.\n            Errors such as uncorrectable ECC, CRC checksum failure, or LBA tag mismatch are included in this field.\n            </summary>\n        </member>\n        <member name=\"F:LibreHardwareMonitor.Interop.Kernel32.NVME_HEALTH_INFO_LOG.NumberErrorInformationLogEntries\">\n            <summary>\n            Number of Error Information Log Entries:  Contains the number of Error Information log entries over the life of the controller\n            </summary>\n        </member>\n        <member name=\"F:LibreHardwareMonitor.Interop.Kernel32.NVME_HEALTH_INFO_LOG.WarningCompositeTemperatureTime\">\n            <summary>\n            Warning Composite Temperature Time:  Contains the amount of time in minutes that the controller is operational and the Composite Temperature is greater than or equal to the Warning Composite\n            Temperature Threshold.\n            </summary>\n        </member>\n        <member name=\"F:LibreHardwareMonitor.Interop.Kernel32.NVME_HEALTH_INFO_LOG.CriticalCompositeTemperatureTime\">\n            <summary>\n            Critical Composite Temperature Time:  Contains the amount of time in minutes that the controller is operational and the Composite Temperature is greater than the Critical Composite Temperature\n            Threshold.\n            </summary>\n        </member>\n        <member name=\"F:LibreHardwareMonitor.Interop.Kernel32.NVME_HEALTH_INFO_LOG.TemperatureSensor\">\n            <summary>\n            Contains the current temperature reported by temperature sensor 1-8.\n            </summary>\n        </member>\n        <member name=\"P:LibreHardwareMonitor.Interop.Kernel32.IOControlCode.Code\">\n            <summary>\n            Gets the resulting IO control code.\n            </summary>\n        </member>\n        <member name=\"M:LibreHardwareMonitor.Interop.Kernel32.IOControlCode.#ctor(System.UInt32,System.UInt32,LibreHardwareMonitor.Interop.Kernel32.IOControlCode.Access)\">\n            <summary>\n            Initializes a new instance of the <see cref=\"T:LibreHardwareMonitor.Interop.Kernel32.IOControlCode\" /> struct.\n            </summary>\n            <param name=\"deviceType\">Type of the device.</param>\n            <param name=\"function\">The function.</param>\n            <param name=\"access\">The access.</param>\n        </member>\n        <member name=\"M:LibreHardwareMonitor.Interop.Kernel32.IOControlCode.#ctor(System.UInt32,System.UInt32,LibreHardwareMonitor.Interop.Kernel32.IOControlCode.Method,LibreHardwareMonitor.Interop.Kernel32.IOControlCode.Access)\">\n            <summary>\n            Initializes a new instance of the <see cref=\"T:LibreHardwareMonitor.Interop.Kernel32.IOControlCode\" /> struct.\n            </summary>\n            <param name=\"deviceType\">Type of the device.</param>\n            <param name=\"function\">The function.</param>\n            <param name=\"method\">The method.</param>\n            <param name=\"access\">The access.</param>\n        </member>\n        <member name=\"F:LibreHardwareMonitor.Interop.NvidiaML.NvmlReturn.Success\">\n            <summary>\n            The operation was successful\n            </summary>\n        </member>\n        <member name=\"F:LibreHardwareMonitor.Interop.NvidiaML.NvmlReturn.Uninitialized\">\n            <summary>\n            NvidiaML was not first initialized with nvmlInit()\n            </summary>\n        </member>\n        <member name=\"F:LibreHardwareMonitor.Interop.NvidiaML.NvmlReturn.InvalidArgument\">\n            <summary>\n            A supplied argument is invalid\n            </summary>\n        </member>\n        <member name=\"F:LibreHardwareMonitor.Interop.NvidiaML.NvmlReturn.NotSupported\">\n            <summary>\n            The requested operation is not available on target device\n            </summary>\n        </member>\n        <member name=\"F:LibreHardwareMonitor.Interop.NvidiaML.NvmlReturn.NoPermission\">\n            <summary>\n            The current user does not have permission for operation\n            </summary>\n        </member>\n        <member name=\"F:LibreHardwareMonitor.Interop.NvidiaML.NvmlReturn.NotFound\">\n            <summary>\n            A query to find an object was unsuccessful\n            </summary>\n        </member>\n        <member name=\"F:LibreHardwareMonitor.Interop.NvidiaML.NvmlReturn.InsufficientSize\">\n            <summary>\n            An input argument is not large enough\n            </summary>\n        </member>\n        <member name=\"F:LibreHardwareMonitor.Interop.NvidiaML.NvmlReturn.InsufficientPower\">\n            <summary>\n            A device's external power cables are not properly attached\n            </summary>\n        </member>\n        <member name=\"F:LibreHardwareMonitor.Interop.NvidiaML.NvmlReturn.DriverNotLoaded\">\n            <summary>\n            NVIDIA driver is not loaded\n            </summary>\n        </member>\n        <member name=\"F:LibreHardwareMonitor.Interop.NvidiaML.NvmlReturn.TimeOut\">\n            <summary>\n            User provided timeout passed\n            </summary>\n        </member>\n        <member name=\"F:LibreHardwareMonitor.Interop.NvidiaML.NvmlReturn.IRQIssue\">\n            <summary>\n            NVIDIA Kernel detected an interrupt issue with a GPU\n            </summary>\n        </member>\n        <member name=\"F:LibreHardwareMonitor.Interop.NvidiaML.NvmlReturn.LibraryNotFound\">\n            <summary>\n            NvidiaML Shared Library couldn't be found or loaded\n            </summary>\n        </member>\n        <member name=\"F:LibreHardwareMonitor.Interop.NvidiaML.NvmlReturn.FunctionNotFound\">\n            <summary>\n            Local version of NvidiaML doesn't implement this function\n            </summary>\n        </member>\n        <member name=\"F:LibreHardwareMonitor.Interop.NvidiaML.NvmlReturn.CorruptedInfoRom\">\n            <summary>\n            infoROM is corrupted\n            </summary>\n        </member>\n        <member name=\"F:LibreHardwareMonitor.Interop.NvidiaML.NvmlReturn.GpuIsLost\">\n            <summary>\n            The GPU has fallen off the bus or has otherwise become inaccessible\n            </summary>\n        </member>\n        <member name=\"F:LibreHardwareMonitor.Interop.NvidiaML.NvmlReturn.ResetRequired\">\n            <summary>\n            The GPU requires a reset before it can be used again\n            </summary>\n        </member>\n        <member name=\"F:LibreHardwareMonitor.Interop.NvidiaML.NvmlReturn.OperatingSystem\">\n            <summary>\n            The GPU control device has been blocked by the operating system/cgroups\n            </summary>\n        </member>\n        <member name=\"F:LibreHardwareMonitor.Interop.NvidiaML.NvmlReturn.LibRmVersionMismatch\">\n            <summary>\n            RM detects a driver/library version mismatch\n            </summary>\n        </member>\n        <member name=\"F:LibreHardwareMonitor.Interop.NvidiaML.NvmlReturn.InUse\">\n            <summary>\n            An operation cannot be performed because the GPU is currently in use\n            </summary>\n        </member>\n        <member name=\"F:LibreHardwareMonitor.Interop.NvidiaML.NvmlReturn.Unknown\">\n            <summary>\n            An public driver error occurred\n            </summary>\n        </member>\n        <member name=\"T:LibreHardwareMonitor.Interop.Ring0\">\n            <summary>\n            Driver with access at kernel level.\n            </summary>\n        </member>\n        <member name=\"T:LibreHardwareMonitor.Interop.WinNt\">\n            <summary>\n            Contains Win32 definitions for Windows NT.\n            </summary>\n        </member>\n        <member name=\"T:LibreHardwareMonitor.Interop.WinNt.LUID\">\n            <summary>\n            Describes a local identifier for an adapter.\n            </summary>\n        </member>\n        <member name=\"T:LibreHardwareMonitor.Interop.WinNt.LARGE_INTEGER\">\n            <summary>\n            Represents a 64-bit signed integer value.\n            </summary>\n        </member>\n        <member name=\"T:LibreHardwareMonitor.Software.OperatingSystem\">\n            <summary>\n            Contains basic information about the operating system.\n            </summary>\n        </member>\n        <member name=\"M:LibreHardwareMonitor.Software.OperatingSystem.#cctor\">\n            <summary>\n            Statically checks if the current system <see cref=\"P:LibreHardwareMonitor.Software.OperatingSystem.Is64Bit\"/> and <see cref=\"P:LibreHardwareMonitor.Software.OperatingSystem.IsUnix\"/>.\n            </summary>\n        </member>\n        <member name=\"P:LibreHardwareMonitor.Software.OperatingSystem.Is64Bit\">\n            <summary>\n            Gets information about whether the current system is 64 bit.\n            </summary>\n        </member>\n        <member name=\"P:LibreHardwareMonitor.Software.OperatingSystem.IsUnix\">\n            <summary>\n            Gets information about whether the current system is Unix based.\n            </summary>\n        </member>\n    </members>\n</doc>\n"
  },
  {
    "path": "OpenHardwareMonitorApi/OpenHardwareMonitorApi.vcxproj",
    "content": "﻿<?xml version=\"1.0\" encoding=\"utf-8\"?>\n<Project DefaultTargets=\"Build\" ToolsVersion=\"15.0\" xmlns=\"http://schemas.microsoft.com/developer/msbuild/2003\">\n  <ItemGroup Label=\"ProjectConfigurations\">\n    <ProjectConfiguration Include=\"Debug|Win32\">\n      <Configuration>Debug</Configuration>\n      <Platform>Win32</Platform>\n    </ProjectConfiguration>\n    <ProjectConfiguration Include=\"Release|Win32\">\n      <Configuration>Release</Configuration>\n      <Platform>Win32</Platform>\n    </ProjectConfiguration>\n    <ProjectConfiguration Include=\"Debug|x64\">\n      <Configuration>Debug</Configuration>\n      <Platform>x64</Platform>\n    </ProjectConfiguration>\n    <ProjectConfiguration Include=\"Release|x64\">\n      <Configuration>Release</Configuration>\n      <Platform>x64</Platform>\n    </ProjectConfiguration>\n  </ItemGroup>\n  <PropertyGroup Label=\"Globals\">\n    <ProjectGuid>{C0A42F4A-ABB3-4575-B4D5-CEDD8379AC26}</ProjectGuid>\n    <TargetFrameworkVersion>v4.5.2</TargetFrameworkVersion>\n    <Keyword>ManagedCProj</Keyword>\n    <RootNamespace>OpenHardwareMonitorApi</RootNamespace>\n    <WindowsTargetPlatformVersion>10.0</WindowsTargetPlatformVersion>\n  </PropertyGroup>\n  <Import Project=\"$(VCTargetsPath)\\Microsoft.Cpp.Default.props\" />\n  <PropertyGroup Condition=\"'$(Configuration)|$(Platform)'=='Debug|Win32'\" Label=\"Configuration\">\n    <ConfigurationType>DynamicLibrary</ConfigurationType>\n    <UseDebugLibraries>true</UseDebugLibraries>\n    <PlatformToolset>v142</PlatformToolset>\n    <CLRSupport>true</CLRSupport>\n    <CharacterSet>Unicode</CharacterSet>\n  </PropertyGroup>\n  <PropertyGroup Condition=\"'$(Configuration)|$(Platform)'=='Release|Win32'\" Label=\"Configuration\">\n    <ConfigurationType>DynamicLibrary</ConfigurationType>\n    <UseDebugLibraries>false</UseDebugLibraries>\n    <PlatformToolset>v142</PlatformToolset>\n    <CLRSupport>true</CLRSupport>\n    <CharacterSet>Unicode</CharacterSet>\n  </PropertyGroup>\n  <PropertyGroup Condition=\"'$(Configuration)|$(Platform)'=='Debug|x64'\" Label=\"Configuration\">\n    <ConfigurationType>DynamicLibrary</ConfigurationType>\n    <UseDebugLibraries>true</UseDebugLibraries>\n    <PlatformToolset>v142</PlatformToolset>\n    <CLRSupport>true</CLRSupport>\n    <CharacterSet>Unicode</CharacterSet>\n  </PropertyGroup>\n  <PropertyGroup Condition=\"'$(Configuration)|$(Platform)'=='Release|x64'\" Label=\"Configuration\">\n    <ConfigurationType>DynamicLibrary</ConfigurationType>\n    <UseDebugLibraries>false</UseDebugLibraries>\n    <PlatformToolset>v142</PlatformToolset>\n    <CLRSupport>true</CLRSupport>\n    <CharacterSet>Unicode</CharacterSet>\n  </PropertyGroup>\n  <Import Project=\"$(VCTargetsPath)\\Microsoft.Cpp.props\" />\n  <ImportGroup Label=\"ExtensionSettings\">\n  </ImportGroup>\n  <ImportGroup Label=\"Shared\">\n  </ImportGroup>\n  <ImportGroup Label=\"PropertySheets\" Condition=\"'$(Configuration)|$(Platform)'=='Debug|Win32'\">\n    <Import Project=\"$(UserRootDir)\\Microsoft.Cpp.$(Platform).user.props\" Condition=\"exists('$(UserRootDir)\\Microsoft.Cpp.$(Platform).user.props')\" Label=\"LocalAppDataPlatform\" />\n  </ImportGroup>\n  <ImportGroup Label=\"PropertySheets\" Condition=\"'$(Configuration)|$(Platform)'=='Release|Win32'\">\n    <Import Project=\"$(UserRootDir)\\Microsoft.Cpp.$(Platform).user.props\" Condition=\"exists('$(UserRootDir)\\Microsoft.Cpp.$(Platform).user.props')\" Label=\"LocalAppDataPlatform\" />\n  </ImportGroup>\n  <ImportGroup Label=\"PropertySheets\" Condition=\"'$(Configuration)|$(Platform)'=='Debug|x64'\">\n    <Import Project=\"$(UserRootDir)\\Microsoft.Cpp.$(Platform).user.props\" Condition=\"exists('$(UserRootDir)\\Microsoft.Cpp.$(Platform).user.props')\" Label=\"LocalAppDataPlatform\" />\n  </ImportGroup>\n  <ImportGroup Label=\"PropertySheets\" Condition=\"'$(Configuration)|$(Platform)'=='Release|x64'\">\n    <Import Project=\"$(UserRootDir)\\Microsoft.Cpp.$(Platform).user.props\" Condition=\"exists('$(UserRootDir)\\Microsoft.Cpp.$(Platform).user.props')\" Label=\"LocalAppDataPlatform\" />\n  </ImportGroup>\n  <PropertyGroup Label=\"UserMacros\" />\n  <PropertyGroup Condition=\"'$(Configuration)|$(Platform)'=='Debug|Win32'\">\n    <LinkIncremental>true</LinkIncremental>\n    <IncludePath>$(ProjectDir)..\\include;$(IncludePath)</IncludePath>\n    <OutDir>$(SolutionDir)Bin\\$(Configuration)\\</OutDir>\n  </PropertyGroup>\n  <PropertyGroup Condition=\"'$(Configuration)|$(Platform)'=='Debug|x64'\">\n    <LinkIncremental>true</LinkIncremental>\n    <IncludePath>$(ProjectDir)..\\include;$(IncludePath)</IncludePath>\n    <OutDir>$(SolutionDir)Bin\\$(Platform)\\$(Configuration)\\</OutDir>\n  </PropertyGroup>\n  <PropertyGroup Condition=\"'$(Configuration)|$(Platform)'=='Release|Win32'\">\n    <LinkIncremental>false</LinkIncremental>\n    <IncludePath>$(ProjectDir)..\\include;$(IncludePath)</IncludePath>\n    <OutDir>$(SolutionDir)Bin\\$(Configuration)\\</OutDir>\n  </PropertyGroup>\n  <PropertyGroup Condition=\"'$(Configuration)|$(Platform)'=='Release|x64'\">\n    <LinkIncremental>false</LinkIncremental>\n    <IncludePath>$(ProjectDir)..\\include;$(IncludePath)</IncludePath>\n    <OutDir>$(SolutionDir)Bin\\$(Platform)\\$(Configuration)\\</OutDir>\n  </PropertyGroup>\n  <ItemDefinitionGroup Condition=\"'$(Configuration)|$(Platform)'=='Debug|Win32'\">\n    <ClCompile>\n      <WarningLevel>Level3</WarningLevel>\n      <Optimization>Disabled</Optimization>\n      <PreprocessorDefinitions>WIN32;_DEBUG;%(PreprocessorDefinitions);OPENHARDWAREMONITOR_EXPORTS</PreprocessorDefinitions>\n      <PrecompiledHeader>Use</PrecompiledHeader>\n    </ClCompile>\n    <Link>\n      <AdditionalDependencies />\n    </Link>\n  </ItemDefinitionGroup>\n  <ItemDefinitionGroup Condition=\"'$(Configuration)|$(Platform)'=='Debug|x64'\">\n    <ClCompile>\n      <WarningLevel>Level3</WarningLevel>\n      <Optimization>Disabled</Optimization>\n      <PreprocessorDefinitions>_DEBUG;%(PreprocessorDefinitions);OPENHARDWAREMONITOR_EXPORTS</PreprocessorDefinitions>\n      <PrecompiledHeader>Use</PrecompiledHeader>\n    </ClCompile>\n    <Link>\n      <AdditionalDependencies />\n    </Link>\n  </ItemDefinitionGroup>\n  <ItemDefinitionGroup Condition=\"'$(Configuration)|$(Platform)'=='Release|Win32'\">\n    <ClCompile>\n      <WarningLevel>Level3</WarningLevel>\n      <PreprocessorDefinitions>WIN32;NDEBUG;%(PreprocessorDefinitions);OPENHARDWAREMONITOR_EXPORTS</PreprocessorDefinitions>\n      <PrecompiledHeader>Use</PrecompiledHeader>\n    </ClCompile>\n    <Link>\n      <AdditionalDependencies />\n    </Link>\n  </ItemDefinitionGroup>\n  <ItemDefinitionGroup Condition=\"'$(Configuration)|$(Platform)'=='Release|x64'\">\n    <ClCompile>\n      <WarningLevel>Level3</WarningLevel>\n      <PreprocessorDefinitions>NDEBUG;%(PreprocessorDefinitions);OPENHARDWAREMONITOR_EXPORTS</PreprocessorDefinitions>\n      <PrecompiledHeader>Use</PrecompiledHeader>\n    </ClCompile>\n    <Link>\n      <AdditionalDependencies />\n    </Link>\n  </ItemDefinitionGroup>\n  <ItemGroup>\n    <Reference Include=\"LibreHardwareMonitorLib\">\n      <HintPath>LibreHardwareMonitorLib.dll</HintPath>\n    </Reference>\n    <Reference Include=\"System\" />\n    <Reference Include=\"System.Data\" />\n    <Reference Include=\"System.Xml\" />\n  </ItemGroup>\n  <ItemGroup>\n    <ClInclude Include=\"..\\include\\OpenHardwareMonitor\\OpenHardwareMonitorApi.h\" />\n    <ClInclude Include=\"..\\include\\OpenHardwareMonitor\\OpenHardwareMonitorGlobal.h\" />\n    <ClInclude Include=\"OpenHardwareMonitorImp.h\" />\n    <ClInclude Include=\"resource.h\" />\n    <ClInclude Include=\"Stdafx.h\" />\n    <ClInclude Include=\"UpdateVisitor.h\" />\n  </ItemGroup>\n  <ItemGroup>\n    <ClCompile Include=\"OpenHardwareMonitorImp.cpp\" />\n    <ClCompile Include=\"Stdafx.cpp\">\n      <PrecompiledHeader Condition=\"'$(Configuration)|$(Platform)'=='Debug|Win32'\">Create</PrecompiledHeader>\n      <PrecompiledHeader Condition=\"'$(Configuration)|$(Platform)'=='Debug|x64'\">Create</PrecompiledHeader>\n      <PrecompiledHeader Condition=\"'$(Configuration)|$(Platform)'=='Release|Win32'\">Create</PrecompiledHeader>\n      <PrecompiledHeader Condition=\"'$(Configuration)|$(Platform)'=='Release|x64'\">Create</PrecompiledHeader>\n    </ClCompile>\n    <ClCompile Include=\"UpdateVisitor.cpp\" />\n  </ItemGroup>\n  <ItemGroup>\n    <Text Include=\"ReadMe.txt\" />\n  </ItemGroup>\n  <ItemGroup>\n    <ResourceCompile Include=\"app.rc\" />\n  </ItemGroup>\n  <Import Project=\"$(VCTargetsPath)\\Microsoft.Cpp.targets\" />\n  <ImportGroup Label=\"ExtensionTargets\">\n  </ImportGroup>\n</Project>"
  },
  {
    "path": "OpenHardwareMonitorApi/OpenHardwareMonitorApi.vcxproj.filters",
    "content": "﻿<?xml version=\"1.0\" encoding=\"utf-8\"?>\n<Project ToolsVersion=\"4.0\" xmlns=\"http://schemas.microsoft.com/developer/msbuild/2003\">\n  <ItemGroup>\n    <Filter Include=\"源文件\">\n      <UniqueIdentifier>{4FC737F1-C7A5-4376-A066-2A32D752A2FF}</UniqueIdentifier>\n      <Extensions>cpp;c;cc;cxx;def;odl;idl;hpj;bat;asm;asmx</Extensions>\n    </Filter>\n    <Filter Include=\"头文件\">\n      <UniqueIdentifier>{93995380-89BD-4b04-88EB-625FBE52EBFB}</UniqueIdentifier>\n      <Extensions>h;hh;hpp;hxx;hm;inl;inc;xsd</Extensions>\n    </Filter>\n    <Filter Include=\"资源文件\">\n      <UniqueIdentifier>{67DA6AB6-F800-4c08-8B7A-83BB121AAD01}</UniqueIdentifier>\n      <Extensions>rc;ico;cur;bmp;dlg;rc2;rct;bin;rgs;gif;jpg;jpeg;jpe;resx;tiff;tif;png;wav;mfcribbon-ms</Extensions>\n    </Filter>\n  </ItemGroup>\n  <ItemGroup>\n    <ClInclude Include=\"Stdafx.h\">\n      <Filter>头文件</Filter>\n    </ClInclude>\n    <ClInclude Include=\"resource.h\">\n      <Filter>头文件</Filter>\n    </ClInclude>\n    <ClInclude Include=\"OpenHardwareMonitorImp.h\">\n      <Filter>头文件</Filter>\n    </ClInclude>\n    <ClInclude Include=\"..\\include\\OpenHardwareMonitor\\OpenHardwareMonitorApi.h\">\n      <Filter>头文件</Filter>\n    </ClInclude>\n    <ClInclude Include=\"UpdateVisitor.h\">\n      <Filter>头文件</Filter>\n    </ClInclude>\n    <ClInclude Include=\"..\\include\\OpenHardwareMonitor\\OpenHardwareMonitorGlobal.h\">\n      <Filter>头文件</Filter>\n    </ClInclude>\n  </ItemGroup>\n  <ItemGroup>\n    <ClCompile Include=\"Stdafx.cpp\">\n      <Filter>源文件</Filter>\n    </ClCompile>\n    <ClCompile Include=\"OpenHardwareMonitorImp.cpp\">\n      <Filter>源文件</Filter>\n    </ClCompile>\n    <ClCompile Include=\"UpdateVisitor.cpp\">\n      <Filter>源文件</Filter>\n    </ClCompile>\n  </ItemGroup>\n  <ItemGroup>\n    <Text Include=\"ReadMe.txt\" />\n  </ItemGroup>\n  <ItemGroup>\n    <ResourceCompile Include=\"app.rc\">\n      <Filter>资源文件</Filter>\n    </ResourceCompile>\n  </ItemGroup>\n</Project>"
  },
  {
    "path": "OpenHardwareMonitorApi/OpenHardwareMonitorImp.cpp",
    "content": "﻿// 这是主 DLL 文件。\n\n#include \"stdafx.h\"\n\n#include \"OpenHardwareMonitorImp.h\"\n#include <vector>\n\nnamespace OpenHardwareMonitorApi\n{\n    static std::wstring error_message;\n\n    //将CRL的String类型转换成C++的std::wstring类型\n    static std::wstring ClrStringToStdWstring(System::String^ str)\n    {\n        if (str == nullptr)\n        {\n            return std::wstring();\n        }\n        else\n        {\n            const wchar_t* chars = (const wchar_t*)(Runtime::InteropServices::Marshal::StringToHGlobalUni(str)).ToPointer();\n            std::wstring os = chars;\n            Runtime::InteropServices::Marshal::FreeHGlobal(IntPtr((void*)chars));\n            return os;\n        }\n    }\n\n\n    std::shared_ptr<IOpenHardwareMonitor> CreateInstance()\n    {\n        std::shared_ptr<IOpenHardwareMonitor> pMonitor;\n        try\n        {\n            MonitorGlobal::Instance()->Init();\n            pMonitor = std::make_shared<COpenHardwareMonitor>();\n        }\n        catch (System::Exception^ e)\n        {\n            error_message = ClrStringToStdWstring(e->Message);\n        }\n        return pMonitor;\n    }\n\n    std::wstring GetErrorMessage()\n    {\n        return error_message;\n    }\n\n    float COpenHardwareMonitor::CpuTemperature()\n    {\n        return m_cpu_temperature;\n    }\n\n    float COpenHardwareMonitor::GpuTemperature()\n    {\n        if (m_gpu_nvidia_temperature >= 0)\n            return m_gpu_nvidia_temperature;\n        else\n            return m_gpu_ati_temperature;\n    }\n\n    float COpenHardwareMonitor::HDDTemperature()\n    {\n        return m_hdd_temperature;\n    }\n\n    float COpenHardwareMonitor::MainboardTemperature()\n    {\n        return m_main_board_temperature;\n    }\n\n    float COpenHardwareMonitor::GpuUsage()\n    {\n        if (m_gpu_nvidia_usage >= 0)\n            return m_gpu_nvidia_usage;\n        else\n            return m_gpu_ati_usage;\n    }\n\n    float COpenHardwareMonitor::CpuFreq()\n    {\n            return m_cpu_freq;\n    }\n\n    const std::map<std::wstring, float>& COpenHardwareMonitor::AllHDDTemperature()\n    {\n        return m_all_hdd_temperature;\n    }\n\n    const std::map<std::wstring, float>& COpenHardwareMonitor::AllCpuTemperature()\n    {\n        return m_all_cpu_temperature;\n    }\n\n    const std::map<std::wstring, float>& COpenHardwareMonitor::AllHDDUsage()\n    {\n        return m_all_hdd_usage;\n    }\n\n    void COpenHardwareMonitor::SetCpuEnable(bool enable)\n    {\n        MonitorGlobal::Instance()->computer->IsCpuEnabled = enable;\n    }\n\n    void COpenHardwareMonitor::SetGpuEnable(bool enable)\n    {\n        MonitorGlobal::Instance()->computer->IsGpuEnabled = enable;\n    }\n\n    void COpenHardwareMonitor::SetHddEnable(bool enable)\n    {\n        MonitorGlobal::Instance()->computer->IsStorageEnabled = enable;\n    }\n\n    void COpenHardwareMonitor::SetMainboardEnable(bool enable)\n    {\n        MonitorGlobal::Instance()->computer->IsMotherboardEnabled = enable;\n    }\n    bool COpenHardwareMonitor::GetCPUFreq(IHardware^ hardware, float& freq) {\n        for (int i = 0; i < hardware->Sensors->Length; i++)\n        {\n            if (hardware->Sensors[i]->SensorType == SensorType::Clock)\n            {\n                String^ name = hardware->Sensors[i]->Name;\n                if (name != L\"Bus Speed\")\n                    m_all_cpu_clock[ClrStringToStdWstring(name)] = Convert::ToDouble(hardware->Sensors[i]->Value);\n            }\n        }\n        float sum{};\n        for (auto i : m_all_cpu_clock)\n            sum += i.second;\n        freq = sum / m_all_cpu_clock.size() / 1000.0;\n        return true;\n    }\n    bool COpenHardwareMonitor::GetHardwareTemperature(IHardware^ hardware, float& temperature)\n    {\n        temperature = -1;\n        std::vector<float> all_temperature;\n        float core_temperature{ -1 };\n        System::String^ temperature_name;\n        switch (hardware->HardwareType)\n        {\n        case HardwareType::Cpu:\n            temperature_name = L\"Core Average\";\n            break;\n        case HardwareType::GpuNvidia: case HardwareType::GpuAmd:\n            temperature_name = L\"GPU Core\";\n            break;\n        default:\n            break;\n        }\n        for (int i = 0; i < hardware->Sensors->Length; i++)\n        {\n            //找到温度传感器\n            if (hardware->Sensors[i]->SensorType == SensorType::Temperature)\n            {\n                float cur_temperture = Convert::ToDouble(hardware->Sensors[i]->Value);\n                all_temperature.push_back(cur_temperture);\n                if (hardware->Sensors[i]->Name == temperature_name) //如果找到了名称为temperature_name的温度传感器，则将温度保存到core_temperature里\n                    core_temperature = cur_temperture;\n            }\n        }\n        if (core_temperature >= 0)\n        {\n            temperature = core_temperature;\n            return true;\n        }\n        if (!all_temperature.empty())\n        {\n            //如果有多个温度传感器，则取平均值\n            float sum{};\n            for (auto i : all_temperature)\n                sum += i;\n            temperature = sum / all_temperature.size();\n            return true;\n        }\n        //如果没有找到温度传感器，则在SubHardware中寻找\n        for (int i = 0; i < hardware->SubHardware->Length; i++)\n        {\n            if (GetHardwareTemperature(hardware->SubHardware[i], temperature))\n                return true;\n        }\n        return false;\n    }\n\n    bool COpenHardwareMonitor::GetCpuTemperature(IHardware^ hardware, float& temperature)\n    {\n        temperature = -1;\n        m_all_cpu_temperature.clear();\n        for (int i = 0; i < hardware->Sensors->Length; i++)\n        {\n            //找到温度传感器\n            if (hardware->Sensors[i]->SensorType == SensorType::Temperature)\n            {\n                String^ name = hardware->Sensors[i]->Name;\n                //保存每个CPU传感器的温度\n                m_all_cpu_temperature[ClrStringToStdWstring(name)] = Convert::ToDouble(hardware->Sensors[i]->Value);\n            }\n        }\n        //计算平均温度\n        if (!m_all_cpu_temperature.empty())\n        {\n            float sum{};\n            for (const auto& item : m_all_cpu_temperature)\n                sum += item.second;\n            temperature = sum / m_all_cpu_temperature.size();\n        }\n        return temperature > 0;\n    }\n\n    bool COpenHardwareMonitor::GetGpuUsage(IHardware^ hardware, float& gpu_usage)\n    {\n        for (int i = 0; i < hardware->Sensors->Length; i++)\n        {\n            //找到负载\n            if (hardware->Sensors[i]->SensorType == SensorType::Load)\n            {\n                if (hardware->Sensors[i]->Name == L\"GPU Core\")\n                {\n                    gpu_usage = Convert::ToDouble(hardware->Sensors[i]->Value);\n                    return true;\n                }\n            }\n        }\n        return false;\n    }\n\n    bool COpenHardwareMonitor::GetHddUsage(IHardware^ hardware, float& hdd_usage)\n    {\n        for (int i = 0; i < hardware->Sensors->Length; i++)\n        {\n            //找到负载\n            if (hardware->Sensors[i]->SensorType == SensorType::Load)\n            {\n                if (hardware->Sensors[i]->Name == L\"Total Activity\")\n                {\n                    hdd_usage = Convert::ToDouble(hardware->Sensors[i]->Value);\n                    return true;\n                }\n            }\n        }\n        return false;\n    }\n\n    COpenHardwareMonitor::COpenHardwareMonitor()\n    {\n        ResetAllValues();\n    }\n\n    COpenHardwareMonitor::~COpenHardwareMonitor()\n    {\n        MonitorGlobal::Instance()->UnInit();\n    }\n\n    void COpenHardwareMonitor::ResetAllValues()\n    {\n        m_cpu_temperature = -1;\n        m_gpu_nvidia_temperature = -1;\n        m_gpu_ati_temperature = -1;\n        m_hdd_temperature = -1;\n        m_main_board_temperature = -1;\n        m_gpu_nvidia_usage = -1;\n        m_gpu_ati_usage = -1;\n        m_all_hdd_temperature.clear();\n        m_all_hdd_usage.clear();\n    }\n\n    void COpenHardwareMonitor::InsertValueToMap(std::map<std::wstring, float>& value_map, const std::wstring& key, float value)\n    {\n        auto iter = value_map.find(key);\n        if (iter == value_map.end())\n        {\n            value_map[key] = value;\n        }\n        else\n        {\n            std::wstring key_exist = iter->first;\n            size_t index = key_exist.rfind(L'#');   //查找字符串是否含有#号\n            if (index != std::wstring::npos)\n            {\n                //取到#号后面的数字，将其加1\n                int num = _wtoi(key_exist.substr(index + 1).c_str());\n                num++;\n                key_exist = key_exist.substr(0, index + 1);\n                key_exist += std::to_wstring(num);\n            }\n            else //没有#号则在末尾添加\" #1\"\n            {\n                key_exist += L\" #1\";\n            }\n            value_map[key_exist] = value;\n        }\n    }\n\n    void COpenHardwareMonitor::GetHardwareInfo()\n    {\n        ResetAllValues();\n        error_message.clear();\n        try\n        {\n            auto computer = MonitorGlobal::Instance()->computer;\n            computer->Accept(MonitorGlobal::Instance()->updateVisitor);\n            for (int i = 0; i < computer->Hardware->Count; i++)\n            {\n                //查找硬件类型\n                switch (computer->Hardware[i]->HardwareType)\n                {\n                case HardwareType::Cpu:\n                    if (m_cpu_temperature < 0)\n                        GetCpuTemperature(computer->Hardware[i], m_cpu_temperature);\n                    GetCPUFreq(computer->Hardware[i], m_cpu_freq);\n                    break;\n                case HardwareType::GpuNvidia:\n                    if (m_gpu_nvidia_temperature < 0)\n                        GetHardwareTemperature(computer->Hardware[i], m_gpu_nvidia_temperature);\n                    if (m_gpu_nvidia_usage < 0)\n                        GetGpuUsage(computer->Hardware[i], m_gpu_nvidia_usage);\n                    break;\n                case HardwareType::GpuAmd:\n                    if (m_gpu_ati_temperature < 0)\n                        GetHardwareTemperature(computer->Hardware[i], m_gpu_ati_temperature);\n                    if (m_gpu_ati_usage < 0)\n                        GetGpuUsage(computer->Hardware[i], m_gpu_ati_usage);\n                    break;\n                case HardwareType::Storage:\n                {\n                    float cur_hdd_temperature = -1;\n                    GetHardwareTemperature(computer->Hardware[i], cur_hdd_temperature);\n                    //m_all_hdd_temperature[ClrStringToStdWstring(computer->Hardware[i]->Name)] = cur_hdd_temperature;\n                    InsertValueToMap(m_all_hdd_temperature, ClrStringToStdWstring(computer->Hardware[i]->Name), cur_hdd_temperature);\n                    float cur_hdd_usage = -1;\n                    GetHddUsage(computer->Hardware[i], cur_hdd_usage);\n                    //m_all_hdd_usage[ClrStringToStdWstring(computer->Hardware[i]->Name)] = cur_hdd_usage;\n                    InsertValueToMap(m_all_hdd_usage, ClrStringToStdWstring(computer->Hardware[i]->Name), cur_hdd_usage);\n                    if (m_hdd_temperature < 0)\n                        m_hdd_temperature = cur_hdd_temperature;\n                }\n                break;\n                case HardwareType::Motherboard:\n                    if (m_main_board_temperature < 0)\n                        GetHardwareTemperature(computer->Hardware[i], m_main_board_temperature);\n                    break;\n                default:\n                    break;\n                }\n            }\n        }\n        catch (System::Exception^ e)\n        {\n            error_message = ClrStringToStdWstring(e->Message);\n        }\n    }\n\n    ////////////////////////////////////////////////////////////////////////////////////\n    MonitorGlobal::MonitorGlobal()\n    {\n\n    }\n\n    MonitorGlobal::~MonitorGlobal()\n    {\n\n    }\n\n    void MonitorGlobal::Init()\n    {\n        updateVisitor = gcnew UpdateVisitor();\n        computer = gcnew Computer();\n        computer->Open();\n    }\n\n    void MonitorGlobal::UnInit()\n    {\n        computer->Close();\n    }\n\n}\n"
  },
  {
    "path": "OpenHardwareMonitorApi/OpenHardwareMonitorImp.h",
    "content": "﻿// OpenHardwareMonitorApi.h\n\n#pragma once\n#include <string>\n#include \"OpenHardwareMonitor/OpenHardwareMonitorApi.h\"\n#include \"UpdateVisitor.h\"\n#include <map>\n\nusing namespace System;\nusing namespace LibreHardwareMonitor::Hardware;\n\nnamespace OpenHardwareMonitorApi {\n\n    public class COpenHardwareMonitor : public IOpenHardwareMonitor\n\t{\n    public:\n        COpenHardwareMonitor();\n        virtual ~COpenHardwareMonitor();\n\n        virtual void GetHardwareInfo() override;\n        virtual float CpuTemperature() override;\n        virtual float GpuTemperature() override;\n        virtual float HDDTemperature() override;\n        virtual float MainboardTemperature() override;\n        virtual float GpuUsage() override;\n        virtual float CpuFreq() override;\n        virtual const std::map<std::wstring, float>& AllHDDTemperature() override;\n        virtual const std::map<std::wstring, float>& AllCpuTemperature() override;\n        virtual const std::map<std::wstring, float>& AllHDDUsage() override;\n\n        virtual void SetCpuEnable(bool enable) override;\n        virtual void SetGpuEnable(bool enable) override;\n        virtual void SetHddEnable(bool enable) override;\n        virtual void SetMainboardEnable(bool enable) override;\n\n    private:\n        bool GetHardwareTemperature(IHardware^ hardware, float& temperature);\n        bool GetCpuTemperature(IHardware^ hardware, float& temperature);\n        bool GetGpuUsage(IHardware^ hardware, float& gpu_usage);\n        bool GetHddUsage(IHardware^ hardware, float& hdd_usage);\n        bool GetCPUFreq(IHardware^ hardware, float& freq);\n        void ResetAllValues();\n        //向map中插入一个数值，如果key已经存在，则自动对新插入的key重命名\n        static void InsertValueToMap(std::map<std::wstring, float>& value_map, const std::wstring& key, float value);\n\n    private:\n        float m_cpu_temperature{};\n        float m_gpu_nvidia_temperature{};\n        float m_gpu_ati_temperature{};\n        float m_hdd_temperature{};\n        float m_main_board_temperature{};\n        float m_gpu_nvidia_usage{};\n        float m_gpu_ati_usage{};\n        float m_cpu_freq{};\n        std::map<std::wstring, float> m_all_hdd_temperature;\n        std::map<std::wstring, float> m_all_cpu_temperature;\n        std::map<std::wstring, float> m_all_cpu_clock;\n        std::map<std::wstring, float> m_all_hdd_usage;\n    };\n\n    //一个单实例类\n    //由于COpenHardwareMonitor是非托管类，不能将托管类的对象作为成员变量，此类用于保存托管类的对象\n    public ref class MonitorGlobal\n    {\n    public:\n        MonitorGlobal();\n        ~MonitorGlobal();\n        static MonitorGlobal^ Instance()\n        {\n            if (m_instance == nullptr)\n            {\n                m_instance = gcnew MonitorGlobal();\n            }\n            return m_instance;\n        }\n\n        void Init();\n        void UnInit();\n\n        Computer^ computer;\n        UpdateVisitor^ updateVisitor{};\n\n    private:\n        static MonitorGlobal^ m_instance{};\n    };\n}\n"
  },
  {
    "path": "OpenHardwareMonitorApi/ReadMe.txt",
    "content": "========================================================================\n    DYNAMIC LINK LIBRARY : OpenHardwareMonitorApi Project Overview\n========================================================================\n\nAppWizard has created this OpenHardwareMonitorApi DLL for you.  \n\nThis file contains a summary of what you will find in each of the files that\nmake up your OpenHardwareMonitorApi application.\n\nOpenHardwareMonitorApi.vcxproj\n    This is the main project file for VC++ projects generated using an Application Wizard. \n    It contains information about the version of Visual C++ that generated the file, and \n    information about the platforms, configurations, and project features selected with the\n    Application Wizard.\n\nOpenHardwareMonitorApi.vcxproj.filters\n    This is the filters file for VC++ projects generated using an Application Wizard. \n    It contains information about the association between the files in your project \n    and the filters. This association is used in the IDE to show grouping of files with\n    similar extensions under a specific node (for e.g. \".cpp\" files are associated with the\n    \"Source Files\" filter).\n\nOpenHardwareMonitorApi.cpp\n    This is the main DLL source file.\n\nOpenHardwareMonitorApi.h\n    This file contains a class declaration.\n\nAssemblyInfo.cpp\n\tContains custom attributes for modifying assembly metadata.\n\n/////////////////////////////////////////////////////////////////////////////\nOther notes:\n\nAppWizard uses \"TODO:\" to indicate parts of the source code you\nshould add to or customize.\n\n/////////////////////////////////////////////////////////////////////////////\n"
  },
  {
    "path": "OpenHardwareMonitorApi/Stdafx.cpp",
    "content": "// stdafx.cpp : ֻ׼ļԴļ\n// OpenHardwareMonitorApi.pch ΪԤͷ\n// stdafx.obj ԤϢ\n\n#include \"stdafx.h\"\n"
  },
  {
    "path": "OpenHardwareMonitorApi/Stdafx.h",
    "content": "// stdafx.h : ׼ϵͳļİļ\n// Ǿʹõĵ\n// ضĿİļ\n\n#pragma once\n\n\n"
  },
  {
    "path": "OpenHardwareMonitorApi/UpdateVisitor.cpp",
    "content": "﻿#include \"stdafx.h\"\n#include \"UpdateVisitor.h\"\n\nnamespace OpenHardwareMonitorApi\n{\n    void UpdateVisitor::VisitComputer(IComputer ^ computer)\n    {\n        computer->Traverse(this);\n    }\n\n    void UpdateVisitor::VisitHardware(IHardware ^ hardware)\n    {\n        hardware->Update();\n        for each (IHardware^ subHardware in hardware->SubHardware)\n        {\n            subHardware->Accept(this);\n        }\n    }\n\n    void UpdateVisitor::VisitSensor(ISensor ^ sensor)\n    {\n    }\n\n    void UpdateVisitor::VisitParameter(IParameter ^ parameter)\n    {\n    }\n\n}\n"
  },
  {
    "path": "OpenHardwareMonitorApi/UpdateVisitor.h",
    "content": "﻿#pragma once\n\nusing namespace LibreHardwareMonitor::Hardware;\n\nnamespace OpenHardwareMonitorApi\n{\n    public ref class UpdateVisitor : IVisitor\n    {\n    public:\n        virtual void VisitComputer(IComputer^ computer);\n        virtual void VisitHardware(IHardware^ hardware);\n        virtual void VisitSensor(ISensor^ sensor);\n        virtual void VisitParameter(IParameter^ parameter);\n    };\n\n}\n"
  },
  {
    "path": "OpenHardwareMonitorApi/resource.h",
    "content": "//{{NO_DEPENDENCIES}}\n// Microsoft Visual C++ generated include file.\n// Used by app.rc\n"
  },
  {
    "path": "PluginDemo/CustomDrawItem.cpp",
    "content": "﻿#include \"pch.h\"\n#include \"CustomDrawItem.h\"\n#include \"DataManager.h\"\n\nconst wchar_t* CCustomDrawItem::GetItemName() const\n{\n    return CDataManager::Instance().StringRes(IDS_CUSTOM_DRAW_ITEM);\n}\n\nconst wchar_t* CCustomDrawItem::GetItemId() const\n{\n    return L\"b4zc373y\";\n}\n\nconst wchar_t* CCustomDrawItem::GetItemLableText() const\n{\n    return L\"\";\n}\n\nconst wchar_t* CCustomDrawItem::GetItemValueText() const\n{\n    return L\"\";\n}\n\nconst wchar_t* CCustomDrawItem::GetItemValueSampleText() const\n{\n    return L\"\";\n}\n\nbool CCustomDrawItem::IsCustomDraw() const\n{\n    return true;\n}\n\nint CCustomDrawItem::GetItemWidth() const\n{\n    return 50;\n}\n\nstatic void DrawLine(CDC* pDC, CPoint point1, CPoint point2, COLORREF color)\n{\n    CPen aPen, * pOldPen;\n    aPen.CreatePen(PS_SOLID, 1, color);\n    pOldPen = pDC->SelectObject(&aPen);\n    pDC->MoveTo(point1);\n    pDC->LineTo(point2);\n    pDC->SelectObject(pOldPen);\n}\n\nvoid CCustomDrawItem::DrawItem(void* hDC, int x, int y, int w, int h, bool dark_mode)\n{\n    //绘图句柄\n    CDC* pDC = CDC::FromHandle((HDC)hDC);\n    //矩形区域\n    CRect rect(CPoint(x, y), CSize(w, h));\n    //设置颜色\n    COLORREF color1{ dark_mode ? RGB(255, 143, 107) : RGB(227, 81, 16) };\n    COLORREF color2{ dark_mode ? RGB(183, 241, 96) : RGB(83, 131, 11) };\n    COLORREF color3{ dark_mode ? RGB(158, 218, 251) : RGB(6, 111, 168) };\n    //显示时、分、秒的矩形区域\n    CRect rect1{ rect }, rect2{ rect }, rect3{ rect };\n    rect1.bottom = rect.top + rect.Height() / 3;\n    rect2 = rect1;\n    rect2.MoveToY(rect1.bottom);\n    rect3.top = rect2.bottom;\n    rect1.DeflateRect(1, 1);\n    rect2.DeflateRect(1, 1);\n    rect3.DeflateRect(1, 1);\n    //根据当前时间计算矩形的宽度\n    SYSTEMTIME& time{ CDataManager::Instance().m_system_time };\n    int hour_width{ static_cast<int>((time.wHour + time.wMinute / 60.0) * w / 24) };\n    int min_width{ static_cast<int>((time.wMinute + time.wSecond / 60.0) * w / 60) };\n    int sec_width{ time.wSecond * w / 60 };\n    rect1.right = rect1.left + hour_width;\n    rect2.right = rect2.left + min_width;\n    rect3.right = rect3.left + sec_width;\n    //填充矩形\n    pDC->FillSolidRect(rect1, color1);\n    pDC->FillSolidRect(rect2, color2);\n    pDC->FillSolidRect(rect3, color3);\n    //绘制刻度\n    COLORREF color_scale{ dark_mode ? RGB(225, 225, 225) : RGB(45, 45, 45) };\n    for (int i{}; i < 24; i++)\n    {\n        int x_pos{ i * w / 24 + rect1.left };\n        DrawLine(pDC, CPoint(x_pos, rect1.top), CPoint(x_pos, i % 6 == 0 ? rect1.bottom : rect1.top + rect1.Height() / 2), color_scale);\n    }\n}\n"
  },
  {
    "path": "PluginDemo/CustomDrawItem.h",
    "content": "﻿#pragma once\n#include \"PluginInterface.h\"\n\nclass CCustomDrawItem : public IPluginItem\n{\n    // 通过 IPluginItem 继承\n    virtual const wchar_t* GetItemName() const override;\n    virtual const wchar_t* GetItemId() const override;\n    virtual const wchar_t* GetItemLableText() const override;\n    virtual const wchar_t* GetItemValueText() const override;\n    virtual const wchar_t* GetItemValueSampleText() const override;\n    virtual bool IsCustomDraw() const override;\n    virtual int GetItemWidth() const override;\n    virtual void DrawItem(void* hDC, int x, int y, int w, int h, bool dark_mode) override;\n};\n"
  },
  {
    "path": "PluginDemo/DataManager.cpp",
    "content": "﻿#include \"pch.h\"\n#include \"DataManager.h\"\n\nCDataManager CDataManager::m_instance;\n\nCDataManager::CDataManager()\n{\n}\n\nCDataManager::~CDataManager()\n{\n    SaveConfig();\n}\n\nCDataManager& CDataManager::Instance()\n{\n    return m_instance;\n}\n\nvoid CDataManager::LoadConfig(const std::wstring& config_dir)\n{\n    //获取模块的路径\n    HMODULE hModule = reinterpret_cast<HMODULE>(&__ImageBase);\n    wchar_t path[MAX_PATH];\n    GetModuleFileNameW(hModule, path, MAX_PATH);\n    std::wstring module_path = path;\n    m_config_path = module_path;\n    if (!config_dir.empty())\n    {\n        size_t index = module_path.find_last_of(L\"\\\\/\");\n        //模块的文件名\n        std::wstring module_file_name = module_path.substr(index + 1);\n        m_config_path = config_dir + module_file_name;\n    }\n    m_config_path += L\".ini\";\n    m_setting_data.show_second = GetPrivateProfileInt(_T(\"config\"), _T(\"show_second\"), 0, m_config_path.c_str());\n    //m_setting_data.show_label_text = GetPrivateProfileInt(_T(\"config\"), _T(\"show_label_text\"), 1, config_path.c_str());\n}\n\nstatic void WritePrivateProfileInt(const wchar_t* app_name, const wchar_t* key_name, int value, const wchar_t* file_path)\n{\n    wchar_t buff[16];\n    swprintf_s(buff, L\"%d\", value);\n    WritePrivateProfileString(app_name, key_name, buff, file_path);\n}\n\nvoid CDataManager::SaveConfig() const\n{\n    WritePrivateProfileInt(_T(\"config\"), _T(\"show_second\"), m_setting_data.show_second, m_config_path.c_str());\n    //WritePrivateProfileInt(_T(\"config\"), _T(\"show_label_text\"), m_setting_data.show_label_text, config_path.c_str());\n}\n\nconst CString& CDataManager::StringRes(UINT id)\n{\n    auto iter = m_string_table.find(id);\n    if (iter != m_string_table.end())\n    {\n        return iter->second;\n    }\n    else\n    {\n        AFX_MANAGE_STATE(AfxGetStaticModuleState());\n        m_string_table[id].LoadString(id);\n        return m_string_table[id];\n    }\n}\n"
  },
  {
    "path": "PluginDemo/DataManager.h",
    "content": "﻿#pragma once\n#include <string>\n#include <map>\n\n#define g_data CDataManager::Instance()\n\nstruct SettingData\n{\n    bool show_second{};\n    //bool show_label_text{};\n};\n\nclass CDataManager\n{\nprivate:\n    CDataManager();\n    ~CDataManager();\n\npublic:\n    static CDataManager& Instance();\n\n    void LoadConfig(const std::wstring& config_dir);\n    void SaveConfig() const;\n    const CString& StringRes(UINT id);      //根据资源id获取一个字符串资源\n\npublic:\n    std::wstring m_cur_time;\n    std::wstring m_cur_date;\n    SYSTEMTIME m_system_time;\n    SettingData m_setting_data;\n\nprivate:\n    static CDataManager m_instance;\n    std::wstring m_config_path;\n    std::map<UINT, CString> m_string_table;\n};\n"
  },
  {
    "path": "PluginDemo/OptionsDlg.cpp",
    "content": "﻿// OptionsDlg.cpp: 实现文件\n//\n\n#include \"pch.h\"\n#include \"PluginDemo.h\"\n#include \"OptionsDlg.h\"\n#include \"afxdialogex.h\"\n\n\n// COptionsDlg 对话框\n\nIMPLEMENT_DYNAMIC(COptionsDlg, CDialog)\n\nCOptionsDlg::COptionsDlg(CWnd* pParent /*=nullptr*/)\n\t: CDialog(IDD_OPTIONS_DIALOG, pParent)\n{\n\n}\n\nCOptionsDlg::~COptionsDlg()\n{\n}\n\nvoid COptionsDlg::DoDataExchange(CDataExchange* pDX)\n{\n\tCDialog::DoDataExchange(pDX);\n}\n\n\nBEGIN_MESSAGE_MAP(COptionsDlg, CDialog)\n    ON_BN_CLICKED(IDC_SHOW_SECOND_CHECK, &COptionsDlg::OnBnClickedShowSecondCheck)\nEND_MESSAGE_MAP()\n\n\n// COptionsDlg 消息处理程序\n\n\nBOOL COptionsDlg::OnInitDialog()\n{\n    CDialog::OnInitDialog();\n\n    // TODO:  在此添加额外的初始化\n    //初始化控件状态\n    CheckDlgButton(IDC_SHOW_SECOND_CHECK, m_data.show_second);\n    //CheckDlgButton(IDC_SHOW_LABEL_CHECK, m_data.show_label_text);\n\n    return TRUE;  // return TRUE unless you set the focus to a control\n                  // 异常: OCX 属性页应返回 FALSE\n}\n\n\nvoid COptionsDlg::OnBnClickedShowSecondCheck()\n{\n    // TODO: 在此添加控件通知处理程序代码\n    m_data.show_second = (IsDlgButtonChecked(IDC_SHOW_SECOND_CHECK) != 0);\n}\n"
  },
  {
    "path": "PluginDemo/OptionsDlg.h",
    "content": "﻿#pragma once\n#include \"DataManager.h\"\n\n// COptionsDlg 对话框\n\nclass COptionsDlg : public CDialog\n{\n\tDECLARE_DYNAMIC(COptionsDlg)\n\npublic:\n\tCOptionsDlg(CWnd* pParent = nullptr);   // 标准构造函数\n\tvirtual ~COptionsDlg();\n\n// 对话框数据\n#ifdef AFX_DESIGN_TIME\n\tenum { IDD = IDD_OPTIONS_DIALOG };\n#endif\n\n    SettingData m_data;\n\nprotected:\n\tvirtual void DoDataExchange(CDataExchange* pDX);    // DDX/DDV 支持\n\n\tDECLARE_MESSAGE_MAP()\npublic:\n    virtual BOOL OnInitDialog();\n    afx_msg void OnBnClickedShowSecondCheck();\n};\n"
  },
  {
    "path": "PluginDemo/PluginDemo.cpp",
    "content": "﻿#include \"pch.h\"\n#include \"PluginDemo.h\"\n#include \"DataManager.h\"\n#include \"OptionsDlg.h\"\n\nCPluginDemo CPluginDemo::m_instance;\n\nCPluginDemo::CPluginDemo()\n{\n}\n\nCPluginDemo& CPluginDemo::Instance()\n{\n    return m_instance;\n}\n\nIPluginItem* CPluginDemo::GetItem(int index)\n{\n    switch (index)\n    {\n    case 0:\n        return &m_system_date;\n    case 1:\n        return &m_system_time;\n    case 2:\n        return &m_custom_draw_item;\n    default:\n        break;\n    }\n    return nullptr;\n}\n\nvoid CPluginDemo::DataRequired()\n{\n    //获取时间和日期\n    SYSTEMTIME& system_time{ CDataManager::Instance().m_system_time };\n    GetLocalTime(&system_time);\n    wchar_t buff[128];\n    swprintf_s(buff, L\"%d/%.2d/%.2d\", system_time.wYear, system_time.wMonth, system_time.wDay);\n    CDataManager::Instance().m_cur_date = buff;\n\n    if (CDataManager::Instance().m_setting_data.show_second)\n        swprintf_s(buff, L\"%.2d:%.2d:%.2d\", system_time.wHour, system_time.wMinute, system_time.wSecond);\n    else\n        swprintf_s(buff, L\"%.2d:%.2d\", system_time.wHour, system_time.wMinute);\n    CDataManager::Instance().m_cur_time = buff;\n}\n\nconst wchar_t* CPluginDemo::GetInfo(PluginInfoIndex index)\n{\n    AFX_MANAGE_STATE(AfxGetStaticModuleState());\n    static CString str;\n    switch (index)\n    {\n    case TMI_NAME:\n        str.LoadString(IDS_PLUGIN_NAME);\n        return str.GetString();\n    case TMI_DESCRIPTION:\n        str.LoadString(IDS_PLUGIN_DESCRIPTION);\n        return str.GetString();\n    case TMI_AUTHOR:\n        return L\"zhongyang219\";\n    case TMI_COPYRIGHT:\n        return L\"Copyright (C) by Zhong Yang 2021\";\n    case TMI_VERSION:\n        return L\"1.0\";\n    case ITMPlugin::TMI_URL:\n        return L\"https://github.com/zhongyang219/TrafficMonitor\";\n        break;\n    default:\n        break;\n    }\n    return L\"\";\n}\n\nITMPlugin::OptionReturn CPluginDemo::ShowOptionsDialog(void* hParent)\n{\n    AFX_MANAGE_STATE(AfxGetStaticModuleState());\n    COptionsDlg dlg(CWnd::FromHandle((HWND)hParent));\n    dlg.m_data = CDataManager::Instance().m_setting_data;\n    if (dlg.DoModal() == IDOK)\n    {\n        CDataManager::Instance().m_setting_data = dlg.m_data;\n        return ITMPlugin::OR_OPTION_CHANGED;\n    }\n    return ITMPlugin::OR_OPTION_UNCHANGED;\n}\n\n\nvoid CPluginDemo::OnExtenedInfo(ExtendedInfoIndex index, const wchar_t* data)\n{\n    switch (index)\n    {\n    case ITMPlugin::EI_CONFIG_DIR:\n        //从配置文件读取配置\n        g_data.LoadConfig(std::wstring(data));\n\n        break;\n    default:\n        break;\n    }\n}\n\nITMPlugin* TMPluginGetInstance()\n{\n    AFX_MANAGE_STATE(AfxGetStaticModuleState());\n    return &CPluginDemo::Instance();\n}\n"
  },
  {
    "path": "PluginDemo/PluginDemo.h",
    "content": "﻿#pragma once\n#include \"PluginInterface.h\"\n#include \"PluginSystemDate.h\"\n#include \"PluginSystemTime.h\"\n#include \"CustomDrawItem.h\"\n\nclass CPluginDemo : public ITMPlugin\n{\nprivate:\n    CPluginDemo();\n\npublic:\n    static CPluginDemo& Instance();\n\n    // 通过 ITMPlugin 继承\n    virtual IPluginItem* GetItem(int index) override;\n    virtual void DataRequired() override;\n    virtual const wchar_t* GetInfo(PluginInfoIndex index) override;\n    virtual OptionReturn ShowOptionsDialog(void* hParent) override;\n    virtual void OnExtenedInfo(ExtendedInfoIndex index, const wchar_t* data) override;\n\nprivate:\n    CPluginSystemDate m_system_date;\n    CPluginSystemTime m_system_time;\n    CCustomDrawItem m_custom_draw_item;\n\n    static CPluginDemo m_instance;\n};\n\n#ifdef __cplusplus\nextern \"C\" {\n#endif\n    __declspec(dllexport) ITMPlugin* TMPluginGetInstance();\n\n#ifdef __cplusplus\n}\n#endif\n"
  },
  {
    "path": "PluginDemo/PluginDemo.rc",
    "content": "// Microsoft Visual C++ generated resource script.\n//\n#include \"resource.h\"\n\n#define APSTUDIO_READONLY_SYMBOLS\n/////////////////////////////////////////////////////////////////////////////\n//\n// Generated from the TEXTINCLUDE 2 resource.\n//\n#include \"winres.h\"\n\n/////////////////////////////////////////////////////////////////////////////\n#undef APSTUDIO_READONLY_SYMBOLS\n\n/////////////////////////////////////////////////////////////////////////////\n// (壬й) resources\n\n#if !defined(AFX_RESOURCE_DLL) || defined(AFX_TARG_CHS)\nLANGUAGE LANG_CHINESE, SUBLANG_CHINESE_SIMPLIFIED\n#pragma code_page(936)\n\n#ifdef APSTUDIO_INVOKED\n/////////////////////////////////////////////////////////////////////////////\n//\n// TEXTINCLUDE\n//\n\n1 TEXTINCLUDE \nBEGIN\n    \"resource.h\\0\"\nEND\n\n2 TEXTINCLUDE \nBEGIN\n    \"#include \"\"winres.h\"\"\\r\\n\"\n    \"\\0\"\nEND\n\n3 TEXTINCLUDE \nBEGIN\n    \"\\r\\n\"\n    \"\\0\"\nEND\n\n#endif    // APSTUDIO_INVOKED\n\n\n/////////////////////////////////////////////////////////////////////////////\n//\n// Dialog\n//\n\nIDD_OPTIONS_DIALOG DIALOGEX 0, 0, 165, 77\nSTYLE DS_SETFONT | DS_MODALFRAME | WS_POPUP | WS_CAPTION | WS_SYSMENU\nCAPTION \"ʱں\"\nFONT 9, \"΢ź\", 400, 0, 0x0\nBEGIN\n    DEFPUSHBUTTON   \"ȷ\",IDOK,50,56,50,14\n    PUSHBUTTON      \"ȡ\",IDCANCEL,108,56,50,14\n    GROUPBOX        \"ʽ\",IDC_STATIC,7,7,151,45\n    CONTROL         \"ʾ\",IDC_SHOW_SECOND_CHECK,\"Button\",BS_AUTOCHECKBOX | WS_TABSTOP,15,21,63,10\nEND\n\n\n/////////////////////////////////////////////////////////////////////////////\n//\n// DESIGNINFO\n//\n\n#ifdef APSTUDIO_INVOKED\nGUIDELINES DESIGNINFO\nBEGIN\n    IDD_OPTIONS_DIALOG, DIALOG\n    BEGIN\n        LEFTMARGIN, 7\n        RIGHTMARGIN, 158\n        TOPMARGIN, 7\n        BOTTOMMARGIN, 70\n    END\nEND\n#endif    // APSTUDIO_INVOKED\n\n\n/////////////////////////////////////////////////////////////////////////////\n//\n// AFX_DIALOG_LAYOUT\n//\n\nIDD_OPTIONS_DIALOG AFX_DIALOG_LAYOUT\nBEGIN\n    0\nEND\n\n\n/////////////////////////////////////////////////////////////////////////////\n//\n// Version\n//\n\nVS_VERSION_INFO VERSIONINFO\n FILEVERSION 1,0,0,1\n PRODUCTVERSION 1,0,0,1\n FILEFLAGSMASK 0x3fL\n#ifdef _DEBUG\n FILEFLAGS 0x1L\n#else\n FILEFLAGS 0x0L\n#endif\n FILEOS 0x40004L\n FILETYPE 0x2L\n FILESUBTYPE 0x0L\nBEGIN\n    BLOCK \"StringFileInfo\"\n    BEGIN\n        BLOCK \"080404b0\"\n        BEGIN\n            VALUE \"CompanyName\", \"By ZhongYang\"\n            VALUE \"FileDescription\", \"Plugin Demo for TrafficMonitor\"\n            VALUE \"FileVersion\", \"1.0.0.1\"\n            VALUE \"InternalName\", \"PluginDe.dll\"\n            VALUE \"LegalCopyright\", \"Copyright (C) 2021 By ZhongYang\"\n            VALUE \"OriginalFilename\", \"PluginDe.dll\"\n            VALUE \"ProductName\", \"Plugin Demo for TrafficMonitor\"\n            VALUE \"ProductVersion\", \"1.0.0.1\"\n        END\n    END\n    BLOCK \"VarFileInfo\"\n    BEGIN\n        VALUE \"Translation\", 0x804, 1200\n    END\nEND\n\n\n/////////////////////////////////////////////////////////////////////////////\n//\n// String Table\n//\n\nSTRINGTABLE\nBEGIN\n    IDS_PLUGIN_NAME         \"TrafficMonitorʾ\"\n    IDS_PLUGIN_DESCRIPTION  \"TrafficMonitorʾΪ߿TrafficMonitorṩ\"\n    IDS_TIME                \"ʱ\"\n    IDS_DATE                \"\"\n    IDS_CUSTOM_DRAW_ITEM    \"Իʾ\"\nEND\n\n#endif    // (壬й) resources\n/////////////////////////////////////////////////////////////////////////////\n\n\n/////////////////////////////////////////////////////////////////////////////\n// Ӣ() resources\n\n#if !defined(AFX_RESOURCE_DLL) || defined(AFX_TARG_ENU)\nLANGUAGE LANG_ENGLISH, SUBLANG_ENGLISH_US\n#pragma code_page(1252)\n\n/////////////////////////////////////////////////////////////////////////////\n//\n// Dialog\n//\n\nIDD_OPTIONS_DIALOG DIALOGEX 0, 0, 165, 74\nSTYLE DS_SETFONT | DS_MODALFRAME | WS_POPUP | WS_CAPTION | WS_SYSMENU\nCAPTION \"Time and date settings\"\nFONT 9, \"Segoe UI\", 400, 0, 0x0\nBEGIN\n    DEFPUSHBUTTON   \"OK\",IDOK,50,53,50,14\n    PUSHBUTTON      \"Cancel\",IDCANCEL,108,53,50,14\n    GROUPBOX        \"Format\",IDC_STATIC,7,7,151,42\n    CONTROL         \"Show seconds\",IDC_SHOW_SECOND_CHECK,\"Button\",BS_AUTOCHECKBOX | WS_TABSTOP,15,21,63,10\nEND\n\n\n/////////////////////////////////////////////////////////////////////////////\n//\n// DESIGNINFO\n//\n\n#ifdef APSTUDIO_INVOKED\nGUIDELINES DESIGNINFO\nBEGIN\n    IDD_OPTIONS_DIALOG, DIALOG\n    BEGIN\n        LEFTMARGIN, 7\n        RIGHTMARGIN, 158\n        TOPMARGIN, 7\n        BOTTOMMARGIN, 67\n    END\nEND\n#endif    // APSTUDIO_INVOKED\n\n\n/////////////////////////////////////////////////////////////////////////////\n//\n// AFX_DIALOG_LAYOUT\n//\n\nIDD_OPTIONS_DIALOG AFX_DIALOG_LAYOUT\nBEGIN\n    0\nEND\n\n\n/////////////////////////////////////////////////////////////////////////////\n//\n// String Table\n//\n\nSTRINGTABLE\nBEGIN\n    IDS_PLUGIN_NAME         \"A sample plug-in for TrafficMonitor.\"\n    IDS_PLUGIN_DESCRIPTION  \"A sample plug-in for TrafficMonitor, providing an example for developers to develop TrafficMonitor plug-ins.\"\n    IDS_TIME                \"Time\"\n    IDS_DATE                \"Date\"\n    IDS_CUSTOM_DRAW_ITEM    \"Custom draw example\"\nEND\n\n#endif    // Ӣ() resources\n/////////////////////////////////////////////////////////////////////////////\n\n\n\n#ifndef APSTUDIO_INVOKED\n/////////////////////////////////////////////////////////////////////////////\n//\n// Generated from the TEXTINCLUDE 3 resource.\n//\n\n\n/////////////////////////////////////////////////////////////////////////////\n#endif    // not APSTUDIO_INVOKED\n\n"
  },
  {
    "path": "PluginDemo/PluginDemo.vcxproj",
    "content": "<?xml version=\"1.0\" encoding=\"utf-8\"?>\n<Project DefaultTargets=\"Build\" xmlns=\"http://schemas.microsoft.com/developer/msbuild/2003\">\n  <ItemGroup Label=\"ProjectConfigurations\">\n    <ProjectConfiguration Include=\"Debug|Win32\">\n      <Configuration>Debug</Configuration>\n      <Platform>Win32</Platform>\n    </ProjectConfiguration>\n    <ProjectConfiguration Include=\"Release|Win32\">\n      <Configuration>Release</Configuration>\n      <Platform>Win32</Platform>\n    </ProjectConfiguration>\n    <ProjectConfiguration Include=\"Debug|x64\">\n      <Configuration>Debug</Configuration>\n      <Platform>x64</Platform>\n    </ProjectConfiguration>\n    <ProjectConfiguration Include=\"Release|x64\">\n      <Configuration>Release</Configuration>\n      <Platform>x64</Platform>\n    </ProjectConfiguration>\n  </ItemGroup>\n  <PropertyGroup Label=\"Globals\">\n    <VCProjectVersion>16.0</VCProjectVersion>\n    <Keyword>Win32Proj</Keyword>\n    <ProjectGuid>{d1ca3ecc-dc32-445a-b734-c4db08d4ba34}</ProjectGuid>\n    <RootNamespace>PluginDemo</RootNamespace>\n    <WindowsTargetPlatformVersion>10.0</WindowsTargetPlatformVersion>\n  </PropertyGroup>\n  <Import Project=\"$(VCTargetsPath)\\Microsoft.Cpp.Default.props\" />\n  <PropertyGroup Condition=\"'$(Configuration)|$(Platform)'=='Debug|Win32'\" Label=\"Configuration\">\n    <ConfigurationType>DynamicLibrary</ConfigurationType>\n    <UseDebugLibraries>true</UseDebugLibraries>\n    <PlatformToolset>v142</PlatformToolset>\n    <CharacterSet>Unicode</CharacterSet>\n    <UseOfMfc>Dynamic</UseOfMfc>\n  </PropertyGroup>\n  <PropertyGroup Condition=\"'$(Configuration)|$(Platform)'=='Release|Win32'\" Label=\"Configuration\">\n    <ConfigurationType>DynamicLibrary</ConfigurationType>\n    <UseDebugLibraries>false</UseDebugLibraries>\n    <PlatformToolset>v142</PlatformToolset>\n    <WholeProgramOptimization>true</WholeProgramOptimization>\n    <CharacterSet>Unicode</CharacterSet>\n    <UseOfMfc>Dynamic</UseOfMfc>\n  </PropertyGroup>\n  <PropertyGroup Condition=\"'$(Configuration)|$(Platform)'=='Debug|x64'\" Label=\"Configuration\">\n    <ConfigurationType>DynamicLibrary</ConfigurationType>\n    <UseDebugLibraries>true</UseDebugLibraries>\n    <PlatformToolset>v142</PlatformToolset>\n    <CharacterSet>Unicode</CharacterSet>\n    <UseOfMfc>Dynamic</UseOfMfc>\n  </PropertyGroup>\n  <PropertyGroup Condition=\"'$(Configuration)|$(Platform)'=='Release|x64'\" Label=\"Configuration\">\n    <ConfigurationType>DynamicLibrary</ConfigurationType>\n    <UseDebugLibraries>false</UseDebugLibraries>\n    <PlatformToolset>v142</PlatformToolset>\n    <WholeProgramOptimization>true</WholeProgramOptimization>\n    <CharacterSet>Unicode</CharacterSet>\n    <UseOfMfc>Dynamic</UseOfMfc>\n  </PropertyGroup>\n  <Import Project=\"$(VCTargetsPath)\\Microsoft.Cpp.props\" />\n  <ImportGroup Label=\"ExtensionSettings\">\n  </ImportGroup>\n  <ImportGroup Label=\"Shared\">\n  </ImportGroup>\n  <ImportGroup Label=\"PropertySheets\" Condition=\"'$(Configuration)|$(Platform)'=='Debug|Win32'\">\n    <Import Project=\"$(UserRootDir)\\Microsoft.Cpp.$(Platform).user.props\" Condition=\"exists('$(UserRootDir)\\Microsoft.Cpp.$(Platform).user.props')\" Label=\"LocalAppDataPlatform\" />\n  </ImportGroup>\n  <ImportGroup Label=\"PropertySheets\" Condition=\"'$(Configuration)|$(Platform)'=='Release|Win32'\">\n    <Import Project=\"$(UserRootDir)\\Microsoft.Cpp.$(Platform).user.props\" Condition=\"exists('$(UserRootDir)\\Microsoft.Cpp.$(Platform).user.props')\" Label=\"LocalAppDataPlatform\" />\n  </ImportGroup>\n  <ImportGroup Label=\"PropertySheets\" Condition=\"'$(Configuration)|$(Platform)'=='Debug|x64'\">\n    <Import Project=\"$(UserRootDir)\\Microsoft.Cpp.$(Platform).user.props\" Condition=\"exists('$(UserRootDir)\\Microsoft.Cpp.$(Platform).user.props')\" Label=\"LocalAppDataPlatform\" />\n  </ImportGroup>\n  <ImportGroup Label=\"PropertySheets\" Condition=\"'$(Configuration)|$(Platform)'=='Release|x64'\">\n    <Import Project=\"$(UserRootDir)\\Microsoft.Cpp.$(Platform).user.props\" Condition=\"exists('$(UserRootDir)\\Microsoft.Cpp.$(Platform).user.props')\" Label=\"LocalAppDataPlatform\" />\n  </ImportGroup>\n  <PropertyGroup Label=\"UserMacros\" />\n  <PropertyGroup Condition=\"'$(Configuration)|$(Platform)'=='Debug|Win32'\">\n    <LinkIncremental>true</LinkIncremental>\n    <IncludePath>$(ProjectDir)..\\include;$(IncludePath)</IncludePath>\n    <OutDir>$(SolutionDir)Bin\\$(Configuration)\\plugins\\</OutDir>\n  </PropertyGroup>\n  <PropertyGroup Condition=\"'$(Configuration)|$(Platform)'=='Release|Win32'\">\n    <LinkIncremental>false</LinkIncremental>\n    <IncludePath>$(ProjectDir)..\\include;$(IncludePath)</IncludePath>\n    <OutDir>$(SolutionDir)Bin\\$(Configuration)\\plugins\\</OutDir>\n  </PropertyGroup>\n  <PropertyGroup Condition=\"'$(Configuration)|$(Platform)'=='Debug|x64'\">\n    <LinkIncremental>true</LinkIncremental>\n    <IncludePath>$(ProjectDir)..\\include;$(IncludePath)</IncludePath>\n    <OutDir>$(SolutionDir)Bin\\$(Platform)\\$(Configuration)\\plugins\\</OutDir>\n  </PropertyGroup>\n  <PropertyGroup Condition=\"'$(Configuration)|$(Platform)'=='Release|x64'\">\n    <LinkIncremental>false</LinkIncremental>\n    <IncludePath>$(ProjectDir)..\\include;$(IncludePath)</IncludePath>\n    <OutDir>$(SolutionDir)Bin\\$(Platform)\\$(Configuration)\\plugins\\</OutDir>\n  </PropertyGroup>\n  <ItemDefinitionGroup Condition=\"'$(Configuration)|$(Platform)'=='Debug|Win32'\">\n    <ClCompile>\n      <WarningLevel>Level3</WarningLevel>\n      <SDLCheck>true</SDLCheck>\n      <PreprocessorDefinitions>WIN32;_DEBUG;PLUGINDEMO_EXPORTS;_WINDOWS;_USRDLL;%(PreprocessorDefinitions)</PreprocessorDefinitions>\n      <ConformanceMode>true</ConformanceMode>\n      <PrecompiledHeader>Use</PrecompiledHeader>\n      <PrecompiledHeaderFile>pch.h</PrecompiledHeaderFile>\n    </ClCompile>\n    <Link>\n      <SubSystem>Windows</SubSystem>\n      <GenerateDebugInformation>true</GenerateDebugInformation>\n      <EnableUAC>false</EnableUAC>\n    </Link>\n  </ItemDefinitionGroup>\n  <ItemDefinitionGroup Condition=\"'$(Configuration)|$(Platform)'=='Release|Win32'\">\n    <ClCompile>\n      <WarningLevel>Level3</WarningLevel>\n      <FunctionLevelLinking>true</FunctionLevelLinking>\n      <IntrinsicFunctions>true</IntrinsicFunctions>\n      <SDLCheck>true</SDLCheck>\n      <PreprocessorDefinitions>WIN32;NDEBUG;PLUGINDEMO_EXPORTS;_WINDOWS;_USRDLL;%(PreprocessorDefinitions)</PreprocessorDefinitions>\n      <ConformanceMode>true</ConformanceMode>\n      <PrecompiledHeader>Use</PrecompiledHeader>\n      <PrecompiledHeaderFile>pch.h</PrecompiledHeaderFile>\n    </ClCompile>\n    <Link>\n      <SubSystem>Windows</SubSystem>\n      <EnableCOMDATFolding>true</EnableCOMDATFolding>\n      <OptimizeReferences>true</OptimizeReferences>\n      <GenerateDebugInformation>true</GenerateDebugInformation>\n      <EnableUAC>false</EnableUAC>\n    </Link>\n  </ItemDefinitionGroup>\n  <ItemDefinitionGroup Condition=\"'$(Configuration)|$(Platform)'=='Debug|x64'\">\n    <ClCompile>\n      <WarningLevel>Level3</WarningLevel>\n      <SDLCheck>true</SDLCheck>\n      <PreprocessorDefinitions>_DEBUG;PLUGINDEMO_EXPORTS;_WINDOWS;_USRDLL;%(PreprocessorDefinitions)</PreprocessorDefinitions>\n      <ConformanceMode>true</ConformanceMode>\n      <PrecompiledHeader>Use</PrecompiledHeader>\n      <PrecompiledHeaderFile>pch.h</PrecompiledHeaderFile>\n    </ClCompile>\n    <Link>\n      <SubSystem>Windows</SubSystem>\n      <GenerateDebugInformation>true</GenerateDebugInformation>\n      <EnableUAC>false</EnableUAC>\n    </Link>\n  </ItemDefinitionGroup>\n  <ItemDefinitionGroup Condition=\"'$(Configuration)|$(Platform)'=='Release|x64'\">\n    <ClCompile>\n      <WarningLevel>Level3</WarningLevel>\n      <FunctionLevelLinking>true</FunctionLevelLinking>\n      <IntrinsicFunctions>true</IntrinsicFunctions>\n      <SDLCheck>true</SDLCheck>\n      <PreprocessorDefinitions>NDEBUG;PLUGINDEMO_EXPORTS;_WINDOWS;_USRDLL;%(PreprocessorDefinitions)</PreprocessorDefinitions>\n      <ConformanceMode>true</ConformanceMode>\n      <PrecompiledHeader>Use</PrecompiledHeader>\n      <PrecompiledHeaderFile>pch.h</PrecompiledHeaderFile>\n    </ClCompile>\n    <Link>\n      <SubSystem>Windows</SubSystem>\n      <EnableCOMDATFolding>true</EnableCOMDATFolding>\n      <OptimizeReferences>true</OptimizeReferences>\n      <GenerateDebugInformation>true</GenerateDebugInformation>\n      <EnableUAC>false</EnableUAC>\n    </Link>\n  </ItemDefinitionGroup>\n  <ItemGroup>\n    <ClInclude Include=\"..\\include\\PluginInterface.h\" />\n    <ClInclude Include=\"CustomDrawItem.h\" />\n    <ClInclude Include=\"DataManager.h\" />\n    <ClInclude Include=\"framework.h\" />\n    <ClInclude Include=\"OptionsDlg.h\" />\n    <ClInclude Include=\"pch.h\" />\n    <ClInclude Include=\"PluginDemo.h\" />\n    <ClInclude Include=\"PluginSystemDate.h\" />\n    <ClInclude Include=\"PluginSystemTime.h\" />\n    <ClInclude Include=\"resource.h\" />\n  </ItemGroup>\n  <ItemGroup>\n    <ClCompile Include=\"CustomDrawItem.cpp\" />\n    <ClCompile Include=\"DataManager.cpp\" />\n    <ClCompile Include=\"OptionsDlg.cpp\" />\n    <ClCompile Include=\"pch.cpp\">\n      <PrecompiledHeader Condition=\"'$(Configuration)|$(Platform)'=='Debug|x64'\">Create</PrecompiledHeader>\n      <PrecompiledHeader Condition=\"'$(Configuration)|$(Platform)'=='Debug|Win32'\">Create</PrecompiledHeader>\n      <PrecompiledHeader Condition=\"'$(Configuration)|$(Platform)'=='Release|Win32'\">Create</PrecompiledHeader>\n      <PrecompiledHeader Condition=\"'$(Configuration)|$(Platform)'=='Release|x64'\">Create</PrecompiledHeader>\n    </ClCompile>\n    <ClCompile Include=\"PluginDemo.cpp\" />\n    <ClCompile Include=\"PluginSystemDate.cpp\" />\n    <ClCompile Include=\"PluginSystemTime.cpp\" />\n  </ItemGroup>\n  <ItemGroup>\n    <ResourceCompile Include=\"PluginDemo.rc\" />\n  </ItemGroup>\n  <Import Project=\"$(VCTargetsPath)\\Microsoft.Cpp.targets\" />\n  <ImportGroup Label=\"ExtensionTargets\">\n  </ImportGroup>\n</Project>"
  },
  {
    "path": "PluginDemo/PluginDemo.vcxproj.filters",
    "content": "﻿<?xml version=\"1.0\" encoding=\"utf-8\"?>\n<Project ToolsVersion=\"4.0\" xmlns=\"http://schemas.microsoft.com/developer/msbuild/2003\">\n  <ItemGroup>\n    <Filter Include=\"源文件\">\n      <UniqueIdentifier>{4FC737F1-C7A5-4376-A066-2A32D752A2FF}</UniqueIdentifier>\n      <Extensions>cpp;c;cc;cxx;c++;cppm;ixx;def;odl;idl;hpj;bat;asm;asmx</Extensions>\n    </Filter>\n    <Filter Include=\"头文件\">\n      <UniqueIdentifier>{93995380-89BD-4b04-88EB-625FBE52EBFB}</UniqueIdentifier>\n      <Extensions>h;hh;hpp;hxx;h++;hm;inl;inc;ipp;xsd</Extensions>\n    </Filter>\n    <Filter Include=\"资源文件\">\n      <UniqueIdentifier>{67DA6AB6-F800-4c08-8B7A-83BB121AAD01}</UniqueIdentifier>\n      <Extensions>rc;ico;cur;bmp;dlg;rc2;rct;bin;rgs;gif;jpg;jpeg;jpe;resx;tiff;tif;png;wav;mfcribbon-ms</Extensions>\n    </Filter>\n  </ItemGroup>\n  <ItemGroup>\n    <ClInclude Include=\"framework.h\">\n      <Filter>头文件</Filter>\n    </ClInclude>\n    <ClInclude Include=\"pch.h\">\n      <Filter>头文件</Filter>\n    </ClInclude>\n    <ClInclude Include=\"PluginSystemTime.h\">\n      <Filter>头文件</Filter>\n    </ClInclude>\n    <ClInclude Include=\"PluginSystemDate.h\">\n      <Filter>头文件</Filter>\n    </ClInclude>\n    <ClInclude Include=\"PluginDemo.h\">\n      <Filter>头文件</Filter>\n    </ClInclude>\n    <ClInclude Include=\"DataManager.h\">\n      <Filter>头文件</Filter>\n    </ClInclude>\n    <ClInclude Include=\"..\\include\\PluginInterface.h\">\n      <Filter>头文件</Filter>\n    </ClInclude>\n    <ClInclude Include=\"resource.h\">\n      <Filter>头文件</Filter>\n    </ClInclude>\n    <ClInclude Include=\"OptionsDlg.h\">\n      <Filter>头文件</Filter>\n    </ClInclude>\n    <ClInclude Include=\"CustomDrawItem.h\">\n      <Filter>头文件</Filter>\n    </ClInclude>\n  </ItemGroup>\n  <ItemGroup>\n    <ClCompile Include=\"pch.cpp\">\n      <Filter>源文件</Filter>\n    </ClCompile>\n    <ClCompile Include=\"PluginSystemTime.cpp\">\n      <Filter>源文件</Filter>\n    </ClCompile>\n    <ClCompile Include=\"PluginSystemDate.cpp\">\n      <Filter>源文件</Filter>\n    </ClCompile>\n    <ClCompile Include=\"DataManager.cpp\">\n      <Filter>源文件</Filter>\n    </ClCompile>\n    <ClCompile Include=\"OptionsDlg.cpp\">\n      <Filter>源文件</Filter>\n    </ClCompile>\n    <ClCompile Include=\"PluginDemo.cpp\">\n      <Filter>源文件</Filter>\n    </ClCompile>\n    <ClCompile Include=\"CustomDrawItem.cpp\">\n      <Filter>源文件</Filter>\n    </ClCompile>\n  </ItemGroup>\n  <ItemGroup>\n    <ResourceCompile Include=\"PluginDemo.rc\">\n      <Filter>资源文件</Filter>\n    </ResourceCompile>\n  </ItemGroup>\n</Project>"
  },
  {
    "path": "PluginDemo/PluginSystemDate.cpp",
    "content": "﻿#include \"pch.h\"\n#include <wchar.h>\n#include \"PluginSystemDate.h\"\n#include \"DataManager.h\"\n\nCPluginSystemDate::CPluginSystemDate()\n{\n}\n\nconst wchar_t* CPluginSystemDate::GetItemName() const\n{\n    return CDataManager::Instance().StringRes(IDS_DATE);\n}\n\nconst wchar_t* CPluginSystemDate::GetItemId() const\n{\n    return L\"o282ffc4\";\n}\n\nconst wchar_t* CPluginSystemDate::GetItemLableText() const\n{\n    return CDataManager::Instance().StringRes(IDS_DATE);\n}\n\nconst wchar_t* CPluginSystemDate::GetItemValueText() const\n{\n    return CDataManager::Instance().m_cur_date.c_str();\n}\n\nconst wchar_t* CPluginSystemDate::GetItemValueSampleText() const\n{\n    return L\"2022/08/08\";\n}\n"
  },
  {
    "path": "PluginDemo/PluginSystemDate.h",
    "content": "﻿#pragma once\n#include \"PluginInterface.h\"\n\nclass CPluginSystemDate : public IPluginItem\n{\npublic:\n    CPluginSystemDate();\n\n    // 通过 IPluginItem 继承\n    virtual const wchar_t* GetItemName() const override;\n    virtual const wchar_t* GetItemId() const override;\n    virtual const wchar_t* GetItemLableText() const override;\n    virtual const wchar_t* GetItemValueText() const override;\n    virtual const wchar_t* GetItemValueSampleText() const override;\n\nprivate:\n};\n"
  },
  {
    "path": "PluginDemo/PluginSystemTime.cpp",
    "content": "﻿#include \"pch.h\"\n#include <wchar.h>\n#include \"PluginSystemTime.h\"\n#include \"DataManager.h\"\n\nCPluginSystemTime::CPluginSystemTime()\n{\n}\n\nconst wchar_t* CPluginSystemTime::GetItemName() const\n{\n    return CDataManager::Instance().StringRes(IDS_TIME);\n}\n\nconst wchar_t* CPluginSystemTime::GetItemId() const\n{\n    return L\"ra1YX2g1\";\n}\n\nconst wchar_t* CPluginSystemTime::GetItemLableText() const\n{\n    return CDataManager::Instance().StringRes(IDS_TIME);\n}\n\nconst wchar_t* CPluginSystemTime::GetItemValueText() const\n{\n    return CDataManager::Instance().m_cur_time.c_str();\n}\n\nconst wchar_t* CPluginSystemTime::GetItemValueSampleText() const\n{\n    if (CDataManager::Instance().m_setting_data.show_second)\n        return L\"12:00:00\";\n    else\n        return L\"12:00\";\n}\n"
  },
  {
    "path": "PluginDemo/PluginSystemTime.h",
    "content": "﻿#pragma once\n#include \"PluginInterface.h\"\n\nclass CPluginSystemTime : public IPluginItem\n{\npublic:\n    CPluginSystemTime();\n\npublic:\n    // 通过 IPluginItem 继承\n    virtual const wchar_t* GetItemName() const override;\n    virtual const wchar_t* GetItemId() const override;\n    virtual const wchar_t* GetItemLableText() const override;\n    virtual const wchar_t* GetItemValueText() const override;\n    virtual const wchar_t* GetItemValueSampleText() const override;\n\nprivate:\n};\n"
  },
  {
    "path": "PluginDemo/framework.h",
    "content": "﻿#pragma once\n\n#define WIN32_LEAN_AND_MEAN             // 从 Windows 头文件中排除极少使用的内容\n//// Windows 头文件\n//#include <windows.h>\n\n#include <SDKDDKVer.h>\n\n#include <afxwin.h>         // MFC 核心组件和标准组件\n#include <afxext.h>         // MFC 扩展\n#include <afxdisp.h>        // MFC 自动化类\n\n#ifdef _UNICODE\n#if defined _M_IX86\n#pragma comment(linker,\"/manifestdependency:\\\"type='win32' name='Microsoft.Windows.Common-Controls' version='6.0.0.0' processorArchitecture='x86' publicKeyToken='6595b64144ccf1df' language='*'\\\"\")\n#elif defined _M_X64\n#pragma comment(linker,\"/manifestdependency:\\\"type='win32' name='Microsoft.Windows.Common-Controls' version='6.0.0.0' processorArchitecture='amd64' publicKeyToken='6595b64144ccf1df' language='*'\\\"\")\n#else\n#pragma comment(linker,\"/manifestdependency:\\\"type='win32' name='Microsoft.Windows.Common-Controls' version='6.0.0.0' processorArchitecture='*' publicKeyToken='6595b64144ccf1df' language='*'\\\"\")\n#endif\n#endif\n"
  },
  {
    "path": "PluginDemo/pch.cpp",
    "content": "﻿// pch.cpp: 与预编译标头对应的源文件\n\n#include \"pch.h\"\n\n// 当使用预编译的头时，需要使用此源文件，编译才能成功。\n"
  },
  {
    "path": "PluginDemo/pch.h",
    "content": "﻿// pch.h: 这是预编译标头文件。\n// 下方列出的文件仅编译一次，提高了将来生成的生成性能。\n// 这还将影响 IntelliSense 性能，包括代码完成和许多代码浏览功能。\n// 但是，如果此处列出的文件中的任何一个在生成之间有更新，它们全部都将被重新编译。\n// 请勿在此处添加要频繁更新的文件，这将使得性能优势无效。\n\n#ifndef PCH_H\n#define PCH_H\n\n// 添加要在此处预编译的标头\n#include \"framework.h\"\n\n#include \"resource.h\"\n\n#endif //PCH_H\n"
  },
  {
    "path": "PluginDemo/resource.h",
    "content": "﻿//{{NO_DEPENDENCIES}}\n// Microsoft Visual C++ 生成的包含文件。\n// 供 PluginDemo.rc 使用\n//\n#define IDD_DIALOG1                     101\n#define IDD_OPTIONS_DIALOG              101\n#define IDS_PLUGIN_NAME                 103\n#define IDS_PLUGIN_DESCRIPTION          104\n#define IDS_TIME                        105\n#define IDS_DATE                        106\n#define IDS_CUSTOM_DRAW_ITEM            107\n#define IDC_SHOW_SECOND_CHECK           1001\n#define IDC_SHOW_LABEL_CHECK            1002\n\n// Next default values for new objects\n// \n#ifdef APSTUDIO_INVOKED\n#ifndef APSTUDIO_READONLY_SYMBOLS\n#define _APS_NEXT_RESOURCE_VALUE        104\n#define _APS_NEXT_COMMAND_VALUE         40001\n#define _APS_NEXT_CONTROL_VALUE         1003\n#define _APS_NEXT_SYMED_VALUE           101\n#endif\n#endif\n"
  },
  {
    "path": "README.md",
    "content": "**简体中文 | [English](./README_en-us.md)**<br>\n[![Badge](https://img.shields.io/badge/link-996.icu-%23FF4D5B.svg?style=flat-square)](https://996.icu/#/en_US)\n[![LICENSE](https://img.shields.io/badge/license-Anti%20996-blue.svg?style=flat-square)](https://github.com/996icu/996.ICU/blob/master/LICENSE)\n[![GitHub Workflow Status](https://img.shields.io/github/workflow/status/zhongyang219/TrafficMonitor/Release%20CI?label=Release%20CI&logo=github&style=flat-square)](https://github.com/zhongyang219/TrafficMonitor/actions?query=workflow:\"Release+CI\")\n[![GitHub release](https://img.shields.io/github/release/zhongyang219/TrafficMonitor.svg?style=flat-square)](https://github.com/zhongyang219/TrafficMonitor/releases/latest)\n\n# TrafficMonitor 简介\nTraffic Monitor是一款用于Windows平台的网速监控悬浮窗软件，可以显示当前网速、CPU及内存利用率，支持嵌入到任务栏显示，支持更换皮肤、历史流量统计等功能。\n\n# 相关链接：\n\n请[点击此处](https://github.com/zhongyang219/TrafficMonitor/releases/latest)下载TrafficMonitor的最新版本。<br>\n备用链接：[百度网盘下载](https://pan.baidu.com/s/15PMt7s-ASpyDwtS__4cUhg) 提取码：`ou0m`<br>\n\n国内用户如果遇到Github下载缓慢的问题，可以[点击此处](https://gitee.com/zhongyang219/TrafficMonitor)转到此项目在Gitee上的页面。\n\n如果遇到问题，请[点击此处](./Help.md)。<br>\n\n你也可以[点击此处](https://github.com/zhongyang219/TrafficMonitor/actions?query=workflow:\"Release+CI\")下载TrafficMonitor的预发行构建版本。\n\n从1.80版本开始，TrafficMonitor加入了温度监控功能，如果你不需要温度监控功能，并且在使用1.80以上版本中遇到了问题，建议下载不含温度监控的版本（Lite版本）。（在Release页面找到文件名包含`Lite`的版本。）\n\nTrafficMonitor依赖于Microsoft Visual C++ 运行环境，如果程序启动时提示“找不到MSVC*.dll”，请点击以下链接下载并安装Microsoft Visual C++ 运行环境。\n\n[最新支持的 Visual C++ 可再发行程序包下载 | Microsoft Docs](https://docs.microsoft.com/zh-CN/cpp/windows/latest-supported-vc-redist?view=msvc-170)\n\n# 版本说明\n\nTrafficMonitor提供了普通版和Lite版两种版本可用。普通版包含了所有的功能，Lite版本则不包含温度监控、显卡利用率、硬盘利用率等硬件监控功能。普通版运行需要管理员权限，而Lite版本则不需要。\n\n如果没有监控温度等硬件信息的需要，建议使用Lite版。\n\n以下是两个版本功能对比。\n\n| 功能                          | 普通版 | Lite版 |\n| ----------------------------- | ------ | ------ |\n| 网速监控                      | ✔      | ✔      |\n| CPU、内存利用率          | ✔      | ✔      |\n| CPU、显卡、硬盘、主板温度监控、CPU频率监控  | ✔      | ❌      |\n| 显卡利用率监控                | ✔      | ❌      |\n| 硬盘利用率监控                | ✔      | ❌      |\n| 网络详细信息                  | ✔      | ✔      |\n| 插件系统                      | ✔      | ✔      |\n| 主窗口更换皮肤                | ✔      | ✔      |\n| 需要管理员权限                | 是     | 否     |\n\n# 主要特性\n\n* 显示当前实现网络传输速率、CPU和内存占用率\n* 如果电脑有多个网卡，支持自动和手动选择网络连接\n* 查看网络详细信息\n* 支持嵌入到任务栏显示\n* 支持更换皮肤和自定义皮肤\n* 历史流量统计\n* 硬件信息监控\n* 插件系统\n# 使用说明\n\n**[点击这里](https://github.com/zhongyang219/TrafficMonitor/wiki)转到Wiki页面查看关于TrafficMonitor的详细说明文档。**\n\n# 截图\n\n主悬浮窗：<br>\n![](./Screenshots/main1.png)<br>\n右键菜单：<br>\n![](./Screenshots/main.png)<br>\n任务栏窗口：<br>\n![](./Screenshots/taskbar.PNG)<br>\n多彩皮肤：<br>\n<img src=\"./Screenshots/skins.PNG\" style=\"zoom:80%;\" /><br>\n\n# 如何使用\n程序启动后在会在屏幕中显示一个显示网速的悬浮窗。在悬浮窗上点击鼠标右键可以弹出右键菜单。\n\nTrafficMonitor支持将信息显示到任务栏。但是TrafficMonitor默认只显示主窗口（悬浮窗），如果需要让它嵌入到任务栏显示，请在右键菜单中选择“显示任务栏窗口”命令。\n\n任务栏窗口支持自定义显示项目，默认情况下只显示网速，如果需要显示CPU和内存利用率，请在任务栏右键菜单中的“显示设置”子菜单下勾选需要显示的项目，如下图所示：\n\n<img src=\"./Screenshots/taskbar_item_settings.png\" style=\"zoom:80%;\" />\n\n# 自定义皮肤\n<img src=\"./Screenshots/selecte_skin.png\" style=\"zoom:80%;\" /><br>\n在主窗口或通知区图标右键菜单上选择“其他功能”——“更换皮肤”可以打开更换皮肤界面。[点击此处](https://github.com/zhongyang219/TrafficMonitorSkin/blob/master/皮肤下载.md)可以下载更多皮肤。用户还可以根据自己的需要编辑自己的皮肤。<br>\n皮肤文件放在程序所在目录的`skins`目录下，每个皮肤被放到单独的文件夹下，文件夹的名称就是皮肤的名称。<br>\n其中`background.bmp`和`background_l.bmp`是背景图片，`skin.ini`是皮肤的配置文件，可以通过`skin.ini`指定文本颜色、字体、皮肤作者、每个项目的大小和位置等信息。<br>\n\n从1.80版本开始增加了xml格式的皮肤配置文件`skin.xml`，只有xml格式的皮肤配置文件才支持温度和显卡使用率显示。\n\n详细的皮肤制作教程请点击以下链接：\n\n[皮肤制作教程 · zhongyang219/TrafficMonitor Wiki (github.com)](https://github.com/zhongyang219/TrafficMonitor/wiki/皮肤制作教程)\n\n推荐使用[皮肤编辑器](https://github.com/zhongyang219/TrafficMonitorSkinEditor/releases)来创建或编辑皮肤。<br>\n\n# 选项设置\n<img src=\"./Screenshots/option.jpg\" style=\"zoom:80%;\" /><br>\n在右键菜单选择“选项...”可以进入选项设置。在选项设置对话框中，可以单独设置主窗口和任务栏窗口的文本颜色、字体、背景颜色、网速单位、显示的文本等。<br>\n在“常规设置”选项卡中，可以设置是否在程序时自动检查更新，以及是否需要在开机是自动运行。可以设置在什么时候需要发出消息通知。<br>\n从1.72版本开始，支持每个项目文本颜色单独设置。勾选“指定每个项目的颜色”后，点击“文本颜色”右边的颜色框，会弹出详细颜色设置的对话框，可以在这里单独指定每个项目的颜色。<br>\n\n# 插件系统\n\n从1.82版本开始增加了插件系统，插件dll必须放在“TrafficMonitor.exe”同级目录的“plugins”目录下。程序启动后，插件会自动加载。你可以在右键菜单“更多功能”——“插件管理”中查看并管理已加载的插件。\n\n关于如何开发TrafficMonitor的说明，请参见[插件开发指南 · zhongyang219/TrafficMonitor Wiki (github.com)](https://github.com/zhongyang219/TrafficMonitor/wiki/插件开发指南)。\n\n要下载TrafficMonitor插件，请[点击这里](https://github.com/zhongyang219/TrafficMonitorPlugins/blob/main/download/plugin_download.md)。\n\n# 关于硬件监控功能\n\n从1.80版本开始，TrafficMonitor加入了硬件监控功能（包括温度监控和显卡使用率监控、CPU频率监控），它使用了第三方开源库[LibreHardwareMonitor](https://github.com/LibreHardwareMonitor/LibreHardwareMonitor)。如果你在使用温度监控功能时遇到了问题，请[点击这里](./Help.md#13-关于trafficmonitor温度监控的问题)。\n\n需要注意的是，温度监控功能默认是关闭的，如果你要使用TrafficMonitor的温度监控功能，请到[“选项设置”-“常规设置”-“硬件监控”](https://github.com/zhongyang219/TrafficMonitor/wiki/选项设置#硬件监控)中开启。\n\n**注意：硬件监控功能（包括温度监控和显卡使用率监控）可能存在一些问题，它可能会占用更多的CPU和内存。据部分用户反馈，开启温度功能后会导致程序崩溃和系统死机等问题，请在知晓以上风险后再决定开启硬件监控功能。否则，请不要使用硬件监控功能。**\n\n# 更新日志\n\n**[点击此处查看更新日志](./UpdateLog/update_log.md)**\n"
  },
  {
    "path": "README_en-us.md",
    "content": "**[简体中文](./README.md) | English**<br>\n[![Badge](https://img.shields.io/badge/link-996.icu-%23FF4D5B.svg?style=flat-square)](https://996.icu/#/en_US)\n[![LICENSE](https://img.shields.io/badge/license-Anti%20996-blue.svg?style=flat-square)](https://github.com/996icu/996.ICU/blob/master/LICENSE)\n[![GitHub Workflow Status](https://img.shields.io/github/workflow/status/zhongyang219/TrafficMonitor/Release%20CI?label=Release%20CI&logo=github&style=flat-square)](https://github.com/zhongyang219/TrafficMonitor/actions?query=workflow:\"Release+CI\")\n[![GitHub release](https://img.shields.io/github/release/zhongyang219/TrafficMonitor.svg?style=flat-square)](https://github.com/zhongyang219/TrafficMonitor/releases/latest)\n\n# TrafficMonitor Introduction\nTrafficMonitor is a network monitoring suspension window software on Windows platform. It can display the current network speed, usage of CPU. It also has the function of displaying in the taskbar, skin changing and historical traffic statistics. \n\n# Related Links\n\nPlease [click here](https://github.com/zhongyang219/TrafficMonitor/releases/latest) to download the latest version of TrafficMonitor. \n\nAlternate link: Download from [Baidu Netdisk](https://pan.baidu.com/s/15PMt7s-ASpyDwtS__4cUhg). Access code: `ou0m`  \n\nIf you encounter any problems, please [click here](./Help_en-us.md).  \n\nYou can also [click here](https://github.com/zhongyang219/TrafficMonitor/actions?query=workflow:\"Release+CI\") to download the pre-release build version of TrafficMonitor.\n\nStarting from version 1.80, the temperature monitoring function has been added. If you do not need the temperature monitoring function and encounter problems with version 1.80 or higher, it is recommended to download the version which is without temperature monitoring (Lite version). (Find the version that contains `Lite` in the file name on Release page.)\n\nTrafficMonitor relies on the Microsoft Visual C++ runtime environment. If the program prompts \"Cannot find MSVC*.dll\" when the program starts, please click the link below to download and install the Microsoft Visual C++ runtime environment.\n\n[Latest supported Visual C++ Redistributable downloads | Microsoft Docs](https://docs.microsoft.com/en-us/cpp/windows/latest-supported-vc-redist?view=msvc-170)\n\n# Version description\n\nTrafficMonitor provides two versions, the standard version and the Lite version. The standard version includes all the functions, while the Lite version does not include hardware monitoring functions such as temperature monitoring, GPU usage, and hard disk usage. The standard version requires administrator to run, while the Lite version does not. \n\nIf there is no need to monitor the temperature and other hardware information, it is recommended to use the Lite version.\n\nThe functions comparison of the two versions is as the following.\n\n| Functions                                              | Standard Version | Lite Version |\n| ------------------------------------------------------ | ---------------- | ------------ |\n| Net speed monitoring                                   | ✔                | ✔            |\n| CPU and memory usage monitoring                        | ✔                | ✔            |\n| CPU, GPU, hard disk, main board temperature, CPU frequency monitoring | ✔                | ❌            |\n| GPU usage monitoring                                   | ✔                | ❌            |\n| Hard disk usage monitoring                             | ✔                | ❌            |\n| Network connection details                             | ✔                | ✔            |\n| Plug-in system                                         | ✔                | ✔            |\n| Change the skin of the main window                     | ✔                | ✔            |\n| Administrator rights required                          | Yes              | No           |\n\n# Main Features\n\n* Displays the current network transfer speed, usage of CPU and memory.\n* Support automatic or manual selection of network connections if there are multiple network adapters.\n* Display the network connection details.\n* Support the window displayed in the taskbar.\n* Support for changing skin and customizing skin.\n* Historical traffic statistics.\n* Hardware information monitoring\n* Plug-in system\n\n# Instructions for use\n\n**[Click here](https://github.com/zhongyang219/TrafficMonitor/wiki) to go to the Wiki page to view detailed documentation on TrafficMonitor (Only Chinese Available).**\n\n# Screen Shots\n\nMain Suspension Window:  \n![](./Screenshots/en_us/main1.png)  \nRight-Click Menu:  \n![](./Screenshots/en_us/main.png)  \nTaskbar Window  \n![](./Screenshots/en_us/taskbar.png)  \nColorful Skins:  \n![](./Screenshots/skins.PNG)  \nChange Skins:  \n![](./Screenshots/en_us/selecte_skin.png)  \nOptions:  \n<img src=\"./Screenshots/en_us/option.jpg\" style=\"zoom:80%;\" />  \n\n# How to Use\n\nAfter the program started, a floating window showing the network speed will be displayed on the screen. Click the right mouse button on the floating window to pop up the right-click menu.\n\nTrafficMonitor supports displaying information on the taskbar. But TrafficMonitor only displays the main window (floating window) by default. If you want to embed it into the taskbar, please select the \"Show taskbar window\" command in the right-click menu.\n\nTaskbar window supports customization of display items. By default, only the network speed is displayed. If you need to display the CPU and memory usage, please check the items that need to be displayed under the \"Display Settings\" submenu in the taskbar right-click menu, as shown in the figure below:\n\n<img src=\"./Screenshots/en_us/taskbar_item_settings.png\" style=\"zoom:80%;\" />\n\n# Plug-in system\n\nThe plug-in system has been added since version 1.82. The plug-in dll must be placed in the \"plugins\" directory of the same level directory of \"TrafficMonitor.exe\". After the program is started, the plug-in will be loaded automatically. You can view and manage the loaded plugins in the context menu \"More Functions\"-\"Plugin Management\".\n\nFor instructions on how to develop the plugin-in of TrafficMonitor, please see [Plugin Development Guide · zhongyang219/TrafficMonitor Wiki (github.com)](https://github.com/zhongyang219/TrafficMonitor/wiki/Plugin-Development-Guide).\n\nTo download the TrafficMonitor plug-in, please [click here](https://github.com/zhongyang219/TrafficMonitorPlugins/blob/main/download/plugin_download.md).\n\n# About the hardware monitoring function\n\nStarting from version 1.80, the hardware monitoring functions (including temperature monitoring, CPU frequency and graphics card usage monitoring) has been added in TrafficMonitor. It relies on the open source library [LibreHardwareMonitor](https://github.com/LibreHardwareMonitor/LibreHardwareMonitor). If you encounter problems during using the temperature monitoring function, please [click here](./Help_en-us.md#13-about-the-temperature-monitoring-of-trafficmonitor). \n\nIt should be noted that the temperature monitoring function is turned off by default. If you want to use the temperature monitoring function of TrafficMonitor, please go to [\"Option Settings\"-\"General Settings\"-\"Hardware Monitoring\"](https://github.com/zhongyang219/TrafficMonitor/wiki/选项设置#硬件监控) to enable it.\n\n**Note: The hardware monitoring function (including temperature monitoring and GPU usage monitoring) may have some problems, which may cause more CPU and memory usage. According to feedback from some users, turning on the temperature function will cause problems such as program crashes and system crashes. Please decide to turn on the hardware monitoring function after you are aware of the above risks. Otherwise, please do not use the hardware monitoring function.**\n\n# Update log\n\n**[Click here to view the update log.](./UpdateLog/update_log_en-us.md)**\n"
  },
  {
    "path": "TrafficMonitor/AboutDlg.cpp",
    "content": "﻿#include \"stdafx.h\"\n#include \"TrafficMonitor.h\"\n#include \"AboutDlg.h\"\n#include \"MessageDlg.h\"\n#include \"DrawCommon.h\"\n\nBEGIN_MESSAGE_MAP(CAboutDlg, CBaseDialog)\n    //ON_STN_CLICKED(IDC_STATIC_DONATE, &CAboutDlg::OnStnClickedStaticDonate)\n    ON_MESSAGE(WM_LINK_CLICKED, &CAboutDlg::OnLinkClicked)\n    ON_WM_PAINT()\n    ON_WM_ERASEBKGND()\n    ON_WM_CTLCOLOR()\nEND_MESSAGE_MAP()\n\nCAboutDlg::CAboutDlg() : CBaseDialog(IDD_ABOUTBOX)\n{\n}\n\nvoid CAboutDlg::DoDataExchange(CDataExchange* pDX)\n{\n    CBaseDialog::DoDataExchange(pDX);\n    DDX_Control(pDX, IDC_STATIC_MAIL, m_mail);\n    DDX_Control(pDX, IDC_STATIC_ACKNOWLEDGEMENT, m_acknowledgement);\n    DDX_Control(pDX, IDC_STATIC_GITHUB, m_github);\n    DDX_Control(pDX, IDC_STATIC_DONATE, m_donate);\n    DDX_Control(pDX, IDC_TRANSLATOR_STATIC, m_translaotr_static);\n    DDX_Control(pDX, IDC_STATIC_LICENSE, m_license);\n    DDX_Control(pDX, IDC_OPENHARDWAREMONITOR_LINK, m_openhardwaremonitor_link);\n    DDX_Control(pDX, IDC_TINYXML2_LINK, m_tinyxml2_link);\n    DDX_Control(pDX, IDC_MUSICPLAYER2_LINK, m_musicplayer2_link);\n    DDX_Control(pDX, IDC_SIMPLENOTEPAD_LINK, m_simplenotepad_link);\n    DDX_Control(pDX, IDC_STATIC_GITEE, m_gitee);\n}\n\nCString CAboutDlg::GetDonateList()\n{\n    return CCommon::GetTextResource(IDR_ACKNOWLEDGEMENT_TEXT, 2);\n}\n\nCString CAboutDlg::GetDialogName() const\n{\n    return _T(\"AboutDlg\");\n}\n\nBOOL CAboutDlg::OnInitDialog()\n{\n    CBaseDialog::OnInitDialog();\n\n    // TODO:  在此添加额外的初始化\n    SetWindowText(CCommon::LoadText(IDS_TITLE_ABOUT));\n    m_mail.SetURL(_T(\"mailto:zhongyang219@hotmail.com\"));   //设置超链接\n    //m_check_update.SetURL(_T(\"http://pan.baidu.com/s/1c1LkPQ4\"));\n    m_github.SetURL(_T(\"https://github.com/zhongyang219/TrafficMonitor\"));\n    m_gitee.SetURL(_T(\"https://gitee.com/zhongyang219/TrafficMonitor\"));\n    m_donate.SetLinkIsURL(false);\n    m_acknowledgement.SetLinkIsURL(false);\n    m_license.SetLinkIsURL(false);\n\n    m_openhardwaremonitor_link.SetURL(_T(\"https://github.com/LibreHardwareMonitor/LibreHardwareMonitor\"));\n    m_tinyxml2_link.SetURL(_T(\"https://github.com/leethomason/tinyxml2\"));\n    m_musicplayer2_link.SetURL(_T(\"https://github.com/zhongyang219/MusicPlayer2\"));\n    m_simplenotepad_link.SetURL(_T(\"https://github.com/zhongyang219/SimpleNotePad\"));\n    m_openhardwaremonitor_link.SetBackgroundColor(GetSysColor(COLOR_WINDOW));\n    m_tinyxml2_link.SetBackgroundColor(GetSysColor(COLOR_WINDOW));\n    m_musicplayer2_link.SetBackgroundColor(GetSysColor(COLOR_WINDOW));\n    m_simplenotepad_link.SetBackgroundColor(GetSysColor(COLOR_WINDOW));\n\n    //设置版本信息\n    CString version_info;\n    GetDlgItemText(IDC_STATIC_VERSION, version_info);\n    CString str_lite;\n#ifdef WITHOUT_TEMPERATURE\n    str_lite = CCommon::LoadText(_T(\" (\"), IDS_WITHOUT_TEMPERATURE, _T(\")\"));\n#endif\n    version_info = CCommon::StringFormat(version_info, { str_lite, VERSION });\n\n#ifdef COMPILE_FOR_WINXP\n    version_info += _T(\" (For WinXP)\");\n#endif // COMPILE_FOR_WINXP\n\n#ifdef _M_X64\n    version_info += _T(\" (x64)\");\n#endif\n\n#ifdef _DEBUG\n    version_info += _T(\" (Debug)\");\n#endif\n\n    SetDlgItemText(IDC_STATIC_VERSION, version_info);\n\n    //设置最后编译日期\n    CString temp_str;\n    GetDlgItemText(IDC_STATIC_COPYRIGHT, temp_str);\n    CString str_compile_time = CCommon::GetLastCompileTime();\n    temp_str.Replace(_T(\"<compile_date>\"), str_compile_time);\n    SetDlgItemText(IDC_STATIC_COPYRIGHT, temp_str);\n\n    m_tool_tip.Create(this, TTS_ALWAYSTIP | TTS_NOPREFIX);\n    m_tool_tip.AddTool(&m_mail, CCommon::LoadText(IDS_SEND_EMAIL_TO_ATHOUR, _T(\"\\r\\nmailto:zhongyang219@hotmail.com\")));\n    //m_tool_tip.AddTool(&m_check_update, _T(\"到百度网盘链接查看是否有更新\\r\\nhttp://pan.baidu.com/s/1c1LkPQ4\"));\n    m_tool_tip.AddTool(&m_github, CCommon::LoadText(IDS_GOTO_GITHUB, _T(\"\\r\\nhttps://github.com/zhongyang219/TrafficMonitor\")));\n    m_tool_tip.AddTool(&m_gitee, CCommon::LoadText(IDS_GOTO_GITEE, _T(\"\\r\\nhttps://gitee.com/zhongyang219/TrafficMonitor\")));\n    m_tool_tip.AddTool(&m_donate, CCommon::LoadText(IDS_DONATE_ATHOUR));\n    m_tool_tip.AddTool(&m_openhardwaremonitor_link, m_openhardwaremonitor_link.GetURL());\n    m_tool_tip.AddTool(&m_tinyxml2_link, m_tinyxml2_link.GetURL());\n    m_tool_tip.AddTool(&m_musicplayer2_link, CCommon::LoadText(IDS_MUSICPLAYER2_DESCRIPTION) + _T(\"\\r\\n\") + m_musicplayer2_link.GetURL());\n    m_tool_tip.AddTool(&m_simplenotepad_link, CCommon::LoadText(IDS_SIMPLENOTEPAD_DESCRIPTION) + _T(\"\\r\\n\") + m_simplenotepad_link.GetURL());\n\n    m_tool_tip.SetDelayTime(300);   //设置延迟\n    m_tool_tip.SetMaxTipWidth(800);\n\n    //设置翻译者信息\n    int language_code;\n    language_code = _ttoi(CCommon::LoadText(IDS_LANGUAGE_CODE));\n    if (language_code == 1 || language_code == 2)       //语言是简体中文和英文时不显示翻译者信息\n        m_translaotr_static.ShowWindow(SW_HIDE);\n    if (language_code == 3)     //显示繁体中文翻译者的信息\n    {\n        m_translaotr_static.SetURL(_T(\"http://mkvq.blogspot.com/\"));\n        m_tool_tip.AddTool(&m_translaotr_static, CCommon::LoadText(IDS_CONTACT_TRANSLATOR, _T(\"\\r\\nhttp://mkvq.blogspot.com/\")));\n    }\n    m_translaotr_static.SetBackgroundColor(GetSysColor(COLOR_WINDOW));\n\n    //设置图片的位置\n    CRect rect;\n    GetClientRect(rect);\n    m_rc_pic = rect;\n    ::GetWindowRect(GetDlgItem(IDC_STATIC_VERSION)->GetSafeHwnd(), rect);\n    ScreenToClient(rect);\n    m_rc_pic.bottom = rect.top - theApp.DPI(6);\n    if (m_rc_pic.Height() <= 0)\n        m_rc_pic.bottom = m_rc_pic.top + theApp.DPI(50);\n\n    //加载图片\n    m_about_pic.LoadBitmap(IDB_ABOUT_BACKGROUND_HD);\n\n    return TRUE;  // return TRUE unless you set the focus to a control\n                  // 异常: OCX 属性页应返回 FALSE\n}\n\nBOOL CAboutDlg::PreTranslateMessage(MSG* pMsg)\n{\n    // TODO: 在此添加专用代码和/或调用基类\n    if (pMsg->message == WM_MOUSEMOVE)\n        m_tool_tip.RelayEvent(pMsg);\n\n    return CBaseDialog::PreTranslateMessage(pMsg);\n}\n\n//void CAboutDlg::OnStnClickedStaticDonate()\n//{\n//  CDonateDlg donateDlg;\n//  donateDlg.DoModal();\n//}\n\nafx_msg LRESULT CAboutDlg::OnLinkClicked(WPARAM wParam, LPARAM lParam)\n{\n    CWnd* pCtrl = (CWnd*)wParam;\n    if (pCtrl == &m_donate)\n    {\n        CDonateDlg donateDlg;\n        donateDlg.DoModal();\n    }\n    else if (pCtrl == &m_acknowledgement)\n    {\n        CString strContent = GetDonateList();\n        //strContent += _T(\"\\r\\n\");\n        //strContent += CCommon::LoadText(IDS_ACKNOWLEDGEMENT_EXPLAIN);\n        CMessageDlg dlg;\n        dlg.SetWindowTitle(CCommon::LoadText(IDS_TITLE_ACKNOWLEDGEMENT));\n        //dlg.SetInfoText(CCommon::LoadText(IDS_THANKS_DONORS));\n        dlg.SetMessageText(strContent);\n        dlg.DoModal();\n    }\n    else if (pCtrl == &m_license)\n    {\n        CMessageDlg dlg;\n        dlg.SetWindowTitle(CCommon::LoadText(IDS_LICENSE));\n        dlg.SetInfoText(CCommon::LoadText(IDS_LICENSE_EXPLAIN));\n        dlg.SetMessageText(CCommon::GetTextResource(IDR_LICENSE, 1));\n        dlg.DoModal();\n    }\n\n    return 0;\n}\n\nvoid CAboutDlg::OnPaint()\n{\n    CPaintDC dc(this); // device context for painting\n                       // TODO: 在此处添加消息处理程序代码\n                       // 不为绘图消息调用 CBaseDialog::OnPaint()\n    CDrawCommon draw;\n    draw.Create(&dc, this);\n    draw.GetDC()->FillSolidRect(m_rc_pic, RGB(161, 200, 255));\n    draw.DrawBitmap(m_about_pic, m_rc_pic.TopLeft(), m_rc_pic.Size(), CDrawCommon::StretchMode::FIT);\n}\n\n\nBOOL CAboutDlg::OnEraseBkgnd(CDC* pDC)\n{\n    // TODO: 在此添加消息处理程序代码和/或调用默认值\n    CRect draw_rect;\n    GetClientRect(draw_rect);\n    pDC->FillSolidRect(draw_rect, GetSysColor(COLOR_WINDOW));\n\n    //绘制白色背景\n    int white_height;       //白色区域的高度\n    CRect rc_copyright{};\n    ::GetWindowRect(GetDlgItem(IDOK)->GetSafeHwnd(), rc_copyright);\n    ScreenToClient(rc_copyright);\n    white_height = rc_copyright.top - theApp.DPI(6);\n\n    //绘制“确定”按钮上方的分割线\n    CRect rc_line{ draw_rect };\n    rc_line.top = white_height;\n    rc_line.bottom = white_height + theApp.DPI(1);\n    pDC->FillSolidRect(rc_line, RGB(210, 210, 210));\n\n    //绘制灰色背景\n    CRect rc_gray{ rc_line };\n    rc_gray.top = rc_line.bottom;\n    rc_gray.bottom = draw_rect.bottom;\n    pDC->FillSolidRect(rc_gray, GetSysColor(COLOR_BTNFACE));\n\n    return TRUE;\n    //return CBaseDialog::OnEraseBkgnd(pDC);\n}\n\n\nHBRUSH CAboutDlg::OnCtlColor(CDC* pDC, CWnd* pWnd, UINT nCtlColor)\n{\n    HBRUSH hbr = CBaseDialog::OnCtlColor(pDC, pWnd, nCtlColor);\n\n    // TODO:  在此更改 DC 的任何特性\n    //去掉static控件的灰色灰色背景\n    UINT ctrl_id = pWnd->GetDlgCtrlID();\n    if (ctrl_id == IDC_STATIC_VERSION || ctrl_id == IDC_STATIC_COPYRIGHT || ctrl_id == IDC_STATIC)\n    {\n        static HBRUSH hBackBrush{};\n        if (hBackBrush == NULL)\n            hBackBrush = CreateSolidBrush(GetSysColor(COLOR_WINDOW));\n        pDC->SetBkColor(GetSysColor(COLOR_WINDOW));\n        return hBackBrush;\n    }\n\n    // TODO:  如果默认的不是所需画笔，则返回另一个画笔\n    return hbr;\n}\n"
  },
  {
    "path": "TrafficMonitor/AboutDlg.h",
    "content": "﻿#pragma once\n#include \"DonateDlg.h\"\n#include \"LinkStatic.h\"\n#include \"BaseDialog.h\"\n\n// 用于应用程序“关于”菜单项的 CAboutDlg 对话框\n\nclass CAboutDlg : public CBaseDialog\n{\npublic:\n    CAboutDlg();\n\n    // 对话框数据\n#ifdef AFX_DESIGN_TIME\n    enum { IDD = IDD_ABOUTBOX };\n#endif\n\nprotected:\n    CLinkStatic m_mail;             //“联系作者”超链接\n    CLinkStatic m_acknowledgement;  //“鸣谢”超链接\n    CLinkStatic m_github;           //“GitHub”超链接\n    CLinkStatic m_gitee;            //“Gitee”超链接\n    CLinkStatic m_donate;           //“捐助”超链接\n    CLinkStatic m_license;          //“开源协议”超链接\n    CToolTipCtrl m_tool_tip;            //鼠标指向时的工具提示\n    CLinkStatic m_translaotr_static;\n    CLinkStatic m_openhardwaremonitor_link;\n    CLinkStatic m_tinyxml2_link;\n    CLinkStatic m_musicplayer2_link;\n    CLinkStatic m_simplenotepad_link;\n\n    CRect m_rc_pic;     //图片的位置\n    CBitmap m_about_pic;\n\n    virtual void DoDataExchange(CDataExchange* pDX);    // DDX/DDV 支持\n    CString GetDonateList();        //从资源文件加载捐赠人员名单\n    virtual CString GetDialogName() const override;\n\n    // 实现\nprotected:\n    DECLARE_MESSAGE_MAP()\npublic:\n    virtual BOOL OnInitDialog();\n    virtual BOOL PreTranslateMessage(MSG* pMsg);\n    //  afx_msg void OnStnClickedStaticDonate();\nprotected:\n    afx_msg LRESULT OnLinkClicked(WPARAM wParam, LPARAM lParam);\npublic:\n    afx_msg void OnPaint();\n    afx_msg BOOL OnEraseBkgnd(CDC* pDC);\n    afx_msg HBRUSH OnCtlColor(CDC* pDC, CWnd* pWnd, UINT nCtlColor);\n};\n"
  },
  {
    "path": "TrafficMonitor/AdapterCommon.cpp",
    "content": "#include \"stdafx.h\"\n#include \"AdapterCommon.h\"\n\n\nCAdapterCommon::CAdapterCommon()\n{\n}\n\n\nCAdapterCommon::~CAdapterCommon()\n{\n}\n\nvoid CAdapterCommon::GetAdapterInfo(vector<NetWorkConection>& adapters)\n{\n\tadapters.clear();\n\tPIP_ADAPTER_INFO pIpAdapterInfo = (PIP_ADAPTER_INFO)new BYTE[sizeof(IP_ADAPTER_INFO)];\t\t//PIP_ADAPTER_INFOṹָ洢Ϣ\n\tunsigned long stSize = sizeof(IP_ADAPTER_INFO);\t\t//õṹС,GetAdaptersInfo\n\tint nRel = GetAdaptersInfo(pIpAdapterInfo, &stSize);\t//GetAdaptersInfo,pIpAdapterInfoָ;stSizeһҲһ\n\n\tif (ERROR_BUFFER_OVERFLOW == nRel)\n\t{\n\t\t//صERROR_BUFFER_OVERFLOW\n\t\t//˵GetAdaptersInfoݵڴռ䲻,ͬʱ䴫stSize,ʾҪĿռС\n\t\t//Ҳ˵ΪʲôstSizeһҲһ\n\t\tdelete[] (BYTE*)pIpAdapterInfo;\t//ͷԭڴռ\n\t\tpIpAdapterInfo = (PIP_ADAPTER_INFO)new BYTE[stSize];\t//ڴռ洢Ϣ\n\t\tnRel = GetAdaptersInfo(pIpAdapterInfo, &stSize);\t\t//ٴεGetAdaptersInfo,pIpAdapterInfoָ\n\t}\n\n\tPIP_ADAPTER_INFO pIpAdapterInfoHead = pIpAdapterInfo;\t//pIpAdapterInfoеһԪصĵַ\n\tif (ERROR_SUCCESS == nRel)\n\t{\n\t\twhile (pIpAdapterInfo)\n\t\t{\n\t\t\tNetWorkConection connection;\n\t\t\tconnection.description = pIpAdapterInfo->Description;\n\t\t\tconnection.ip_address = CCommon::StrToUnicode(pIpAdapterInfo->IpAddressList.IpAddress.String);\n\t\t\tconnection.subnet_mask = CCommon::StrToUnicode(pIpAdapterInfo->IpAddressList.IpMask.String);\n\t\t\tconnection.default_gateway = CCommon::StrToUnicode(pIpAdapterInfo->GatewayList.IpAddress.String);\n\n\t\t\tadapters.push_back(connection);\n\t\t\tpIpAdapterInfo = pIpAdapterInfo->Next;\n\t\t}\n\t}\n\t//ͷڴռ\n\tif (pIpAdapterInfoHead)\n\t{\n\t\tdelete[] (BYTE*)pIpAdapterInfoHead;\n\t}\n\tif (adapters.empty())\n\t{\n\t\tNetWorkConection connection{};\n\t\tconnection.description = CCommon::UnicodeToStr(CCommon::LoadText(L\"<\", IDS_NO_CONNECTION, L\">\"));\n\t\tadapters.push_back(connection);\n\t}\n}\n\nvoid CAdapterCommon::RefreshIpAddress(vector<NetWorkConection>& adapters)\n{\n\tvector<NetWorkConection> adapters_tmp;\n\tGetAdapterInfo(adapters_tmp);\n\tfor (const auto& adapter_tmp : adapters_tmp)\n\t{\n\t\tfor (auto& adapter : adapters)\n\t\t{\n\t\t\tif (adapter_tmp.description == adapter.description)\n\t\t\t{\n\t\t\t\tadapter.ip_address = adapter_tmp.ip_address;\n\t\t\t\tadapter.subnet_mask = adapter_tmp.subnet_mask;\n\t\t\t\tadapter.default_gateway = adapter_tmp.default_gateway;\n\t\t\t}\n\t\t}\n\t}\n}\n\nvoid CAdapterCommon::GetIfTableInfo(vector<NetWorkConection>& adapters, MIB_IFTABLE* pIfTable)\n{\n\t//IfTableÿ\n\tfor (size_t i{}; i < adapters.size(); i++)\n\t{\n\t\tif (adapters[i].description.empty())\n\t\t\tcontinue;\n\t\tint index;\n\t\tindex = FindConnectionInIfTable(adapters[i].description, pIfTable);\n\t\tif (index == -1)\t\t//ʹþȷƥķʽûҵģƥķʽٲһ\n\t\t\tindex = FindConnectionInIfTableFuzzy(adapters[i].description, pIfTable);\n\t\t//if (index != -1)\n\t\t//{\n\t\tadapters[i].index = index;\n\t\tadapters[i].in_bytes = pIfTable->table[index].dwInOctets;\n\t\tadapters[i].out_bytes = pIfTable->table[index].dwOutOctets;\n\t\tadapters[i].description_2 = (const char*)pIfTable->table[index].bDescr;\n\t\t//}\n\t}\n}\n\nvoid CAdapterCommon::GetAllIfTableInfo(vector<NetWorkConection>& adapters, MIB_IFTABLE * pIfTable)\n{\n\tvector<NetWorkConection> adapters_tmp;\n\tGetAdapterInfo(adapters_tmp);\t\t//ȡIPַ\n\tadapters.clear();\n\tfor (size_t i{}; i < pIfTable->dwNumEntries; i++)\n\t{\n\t\tNetWorkConection connection;\n\t\tconnection.description = connection.description_2 = (const char*)pIfTable->table[i].bDescr;\n\t\tconnection.index = i;\n\t\tconnection.in_bytes = pIfTable->table[i].dwInOctets;\n\t\tconnection.out_bytes = pIfTable->table[i].dwOutOctets;\n\t\tfor (size_t j{}; j < adapters_tmp.size(); j++)\n\t\t{\n\t\t\tif (connection.description.find(adapters_tmp[j].description) != string::npos)\n\t\t\t{\n\t\t\t\tconnection.ip_address = adapters_tmp[j].ip_address;\n\t\t\t\tconnection.subnet_mask = adapters_tmp[j].subnet_mask;\n\t\t\t\tconnection.default_gateway = adapters_tmp[j].default_gateway;\n\t\t\t\tbreak;\n\t\t\t}\n\t\t}\n\t\tadapters.push_back(connection);\n\t}\n}\n\nint CAdapterCommon::FindConnectionInIfTable(string connection, MIB_IFTABLE* pIfTable)\n{\n\tfor (size_t i{}; i < pIfTable->dwNumEntries; i++)\n\t{\n\t\tstring descr = (const char*)pIfTable->table[i].bDescr;\n\t\tif (descr == connection)\n\t\t\treturn i;\n\t}\n\treturn -1;\n}\n\nint CAdapterCommon::FindConnectionInIfTableFuzzy(string connection, MIB_IFTABLE* pIfTable)\n{\n\tfor (size_t i{}; i < pIfTable->dwNumEntries; i++)\n\t{\n\t\tstring descr = (const char*)pIfTable->table[i].bDescr;\n\t\tsize_t index;\n\t\t//ڽϳַҽ϶̵ַ\n\t\tif (descr.size() >= connection.size())\n\t\t\tindex = descr.find(connection);\n\t\telse\n\t\t\tindex = connection.find(descr);\n\t\tif (index != wstring::npos)\n\t\t\treturn i;\n\t}\n\t//ûҵʹַƥ㷨\n\tdouble max_degree{};\n\tint best_index{};\n\tfor (size_t i{}; i < pIfTable->dwNumEntries; i++)\n\t{\n\t\tstring descr = (const char*)pIfTable->table[i].bDescr;\n\t\tdouble degree = CCommon::StringSimilarDegree_LD(descr, connection);\n\t\tif (degree > max_degree)\n\t\t{\n\t\t\tmax_degree = degree;\n\t\t\tbest_index = i;\n\t\t}\n\t}\n\treturn best_index;\n}\n"
  },
  {
    "path": "TrafficMonitor/AdapterCommon.h",
    "content": "#pragma once\n#include \"Common.h\"\n\n//һϢ\nstruct NetWorkConection\n{\n\tint index{};\t\t\t//MIB_IFTABLEе\n\tstring description;\t\t//ȡGetAdapterInfo\n\tstring description_2;\t//ȡGetIfTable\n\tunsigned int in_bytes;\t//ʼʱѽֽ\n\tunsigned int out_bytes;\t//ʼʱѷֽ\n\twstring ip_address{ L\"-.-.-.-\" };\t//IPַ\n\twstring subnet_mask{ L\"-.-.-.-\" };\t//\n\twstring default_gateway{ L\"-.-.-.-\" };\t//Ĭ\n};\n\nclass CAdapterCommon\n{\npublic:\n\tCAdapterCommon();\n\t~CAdapterCommon();\n\n\t//ȡбIPַ롢ĬϢ\n\tstatic void GetAdapterInfo(vector<NetWorkConection>& adapters);\n\n\t//ˢбеIPַ롢ĬϢ\n\tstatic void RefreshIpAddress(vector<NetWorkConection>& adapters);\n\n\t//ȡбÿӵMIB_IFTABLEеʼʱѽ/ֽϢ\n\tstatic void GetIfTableInfo(vector<NetWorkConection>& adapters, MIB_IFTABLE* pIfTable);\n\n\t//ֱӽMIB_IFTABLEеӵadapters\n\tstatic void GetAllIfTableInfo(vector<NetWorkConection>& adapters, MIB_IFTABLE* pIfTable);\nprivate:\n\t//һжǷIfTableбҲ򷵻-1\n\tstatic int FindConnectionInIfTable(string connection, MIB_IFTABLE* pIfTable);\n\n\t//һжǷIfTableбҲ򷵻-1ֻҪƥ\n\tstatic int FindConnectionInIfTableFuzzy(string connection, MIB_IFTABLE* pIfTable);\n};\n\n"
  },
  {
    "path": "TrafficMonitor/AppAlreadyRuningDlg.cpp",
    "content": "﻿// AppAlreadyRuningDlg.cpp: 实现文件\n//\n\n#include \"stdafx.h\"\n#include \"TrafficMonitor.h\"\n#include \"AppAlreadyRuningDlg.h\"\n#include \"afxdialogex.h\"\n\n\n// CAppAlreadyRuningDlg 对话框\n\nIMPLEMENT_DYNAMIC(CAppAlreadyRuningDlg, CDialog)\n\nCAppAlreadyRuningDlg::CAppAlreadyRuningDlg(HWND handel, CWnd* pParent /*=nullptr*/)\n    : CDialog(IDD_APP_ALREAD_RUNING_DIALOG, pParent), m_handle(handel)\n{\n\n}\n\nCAppAlreadyRuningDlg::~CAppAlreadyRuningDlg()\n{\n}\n\nvoid CAppAlreadyRuningDlg::DoDataExchange(CDataExchange* pDX)\n{\n    CDialog::DoDataExchange(pDX);\n}\n\n\nBEGIN_MESSAGE_MAP(CAppAlreadyRuningDlg, CDialog)\n    ON_BN_CLICKED(IDC_EXIT_INST_BUTTON, &CAppAlreadyRuningDlg::OnBnClickedExitInstButton)\n    ON_BN_CLICKED(IDC_OPEN_SETTINGS_BUTTON, &CAppAlreadyRuningDlg::OnBnClickedOpenSettingsButton)\n    ON_BN_CLICKED(IDC_SHOW_HIDE_MAIN_WINDOW_BUTTON, &CAppAlreadyRuningDlg::OnBnClickedShowHideMainWindowButton)\n    ON_BN_CLICKED(IDC_SHOW_HIDE_TASKBAR_WINDOW_BUTTON, &CAppAlreadyRuningDlg::OnBnClickedShowHideTaskbarWindowButton)\nEND_MESSAGE_MAP()\n\n\n// CAppAlreadyRuningDlg 消息处理程序\n\n\nBOOL CAppAlreadyRuningDlg::OnInitDialog()\n{\n    CDialog::OnInitDialog();\n\n    // TODO:  在此添加额外的初始化\n    SetIcon(AfxGetApp()->LoadIcon(IDR_MAINFRAME), FALSE);\n\n    return TRUE;  // return TRUE unless you set the focus to a control\n                  // 异常: OCX 属性页应返回 FALSE\n}\n\n\nvoid CAppAlreadyRuningDlg::OnBnClickedExitInstButton()\n{\n    ::PostMessage(m_handle, WM_COMMAND, ID_APP_EXIT, 0);\n}\n\n\nvoid CAppAlreadyRuningDlg::OnBnClickedOpenSettingsButton()\n{\n    ::PostMessage(m_handle, WM_COMMAND, ID_OPTIONS, 0);\n}\n\n\nvoid CAppAlreadyRuningDlg::OnBnClickedShowHideMainWindowButton()\n{\n    ::PostMessage(m_handle, WM_COMMAND, ID_SHOW_MAIN_WND, 0);\n}\n\n\nvoid CAppAlreadyRuningDlg::OnBnClickedShowHideTaskbarWindowButton()\n{\n    ::PostMessage(m_handle, WM_COMMAND, ID_SHOW_TASK_BAR_WND, 0);\n}\n"
  },
  {
    "path": "TrafficMonitor/AppAlreadyRuningDlg.h",
    "content": "﻿#pragma once\n\n\n// CAppAlreadyRuningDlg 对话框\n\nclass CAppAlreadyRuningDlg : public CDialog\n{\n    DECLARE_DYNAMIC(CAppAlreadyRuningDlg)\n\npublic:\n    CAppAlreadyRuningDlg(HWND handel, CWnd* pParent = nullptr);   // 标准构造函数\n    virtual ~CAppAlreadyRuningDlg();\n\n    // 对话框数据\n#ifdef AFX_DESIGN_TIME\n    enum { IDD = IDD_APP_ALREAD_RUNING_DIALOG };\n#endif\n\nprivate:\n    HWND m_handle{};        //正在运行的TrafficMonitor进程主窗口的句柄\n\nprotected:\n    virtual void DoDataExchange(CDataExchange* pDX);    // DDX/DDV 支持\n\n    DECLARE_MESSAGE_MAP()\npublic:\n    virtual BOOL OnInitDialog();\n    afx_msg void OnBnClickedExitInstButton();\n    afx_msg void OnBnClickedOpenSettingsButton();\n    afx_msg void OnBnClickedShowHideMainWindowButton();\n    afx_msg void OnBnClickedShowHideTaskbarWindowButton();\n};\n"
  },
  {
    "path": "TrafficMonitor/BaseDialog.cpp",
    "content": "﻿// BaseDialog.cpp : 实现文件\n//\n\n#include \"stdafx.h\"\n#include \"BaseDialog.h\"\n#include \"afxdialogex.h\"\n#include \"IniHelper.h\"\n#include \"TrafficMonitor.h\"\n\n// CBaseDialog 对话框\nstd::map<CString, HWND> CBaseDialog::m_unique_hwnd;\n\nIMPLEMENT_DYNAMIC(CBaseDialog, CDialog)\n\nCBaseDialog::CBaseDialog(UINT nIDTemplate, CWnd* pParent /*=NULL*/)\n    : CDialog(nIDTemplate, pParent)\n{\n\n}\n\nCBaseDialog::~CBaseDialog()\n{\n}\n\nvoid CBaseDialog::SetMinSize(int cx, int cy)\n{\n    m_min_size.cx = cx;\n    m_min_size.cy = cy;\n}\n\nHWND CBaseDialog::GetUniqueHandel(LPCTSTR dlg_name)\n{\n    return m_unique_hwnd[dlg_name];\n}\n\nconst std::map<CString, HWND>& CBaseDialog::AllUniqueHandels()\n{\n    return m_unique_hwnd;\n}\n\nvoid CBaseDialog::LoadConfig()\n{\n    if (!GetDialogName().IsEmpty())\n    {\n        CIniHelper ini{ theApp.m_config_path };\n        //载入窗口大小设置\n        m_window_size.cx = ini.GetInt(_T(\"window_size\"), GetDialogName() + _T(\"_width\"), -1);\n        m_window_size.cy = ini.GetInt(_T(\"window_size\"), GetDialogName() + _T(\"_height\"), -1);\n    }\n}\n\nvoid CBaseDialog::SaveConfig() const\n{\n    if (!GetDialogName().IsEmpty())\n    {\n        CIniHelper ini{ theApp.m_config_path };\n        //保存窗口大小设置\n        ini.WriteInt(_T(\"window_size\"), GetDialogName() + _T(\"_width\"), m_window_size.cx);\n        ini.WriteInt(_T(\"window_size\"), GetDialogName() + _T(\"_height\"), m_window_size.cy);\n        ini.Save();\n    }\n}\n\nvoid CBaseDialog::EnableDlgCtrl(UINT id, bool enable)\n{\n    CWnd* pWnd = GetDlgItem(id);\n    if (pWnd != nullptr)\n        pWnd->EnableWindow(enable);\n}\n\nvoid CBaseDialog::DoDataExchange(CDataExchange* pDX)\n{\n    CDialog::DoDataExchange(pDX);\n}\n\n\nBEGIN_MESSAGE_MAP(CBaseDialog, CDialog)\n    ON_WM_DESTROY()\n    ON_WM_GETMINMAXINFO()\n    ON_WM_SIZE()\nEND_MESSAGE_MAP()\n\n\n// CBaseDialog 消息处理程序\n\n\nBOOL CBaseDialog::OnInitDialog()\n{\n    m_unique_hwnd[GetDialogName()] = m_hWnd;\n    CDialog::OnInitDialog();\n\n    // TODO:  在此添加额外的初始化\n\n    //获取初始时窗口的大小\n    if (m_min_size.cx <= 0 || m_min_size.cy <= 0)\n    {\n        CRect rect;\n        GetWindowRect(rect);\n        m_min_size.cx = rect.Width() * 96 / theApp.GetDpi();\n        m_min_size.cy = rect.Height() * 96 / theApp.GetDpi();\n    }\n\n    //载入设置\n    LoadConfig();\n\n    //初始化窗口大小\n    if (m_window_size.cx > 0 && m_window_size.cy > 0)\n    {\n        SetWindowPos(nullptr, 0, 0, m_window_size.cx, m_window_size.cy, SWP_NOZORDER | SWP_NOMOVE);\n    }\n\n    return TRUE;  // return TRUE unless you set the focus to a control\n                  // 异常: OCX 属性页应返回 FALSE\n}\n\n\nvoid CBaseDialog::OnDestroy()\n{\n    CDialog::OnDestroy();\n\n    // TODO: 在此处添加消息处理程序代码\n    m_unique_hwnd[GetDialogName()] = NULL;\n    SaveConfig();\n}\n\n\nvoid CBaseDialog::OnGetMinMaxInfo(MINMAXINFO* lpMMI)\n{\n    // TODO: 在此添加消息处理程序代码和/或调用默认值\n    //限制窗口最小大小\n    lpMMI->ptMinTrackSize.x = theApp.DPI(m_min_size.cx);\t\t//设置最小宽度\n    lpMMI->ptMinTrackSize.y = theApp.DPI(m_min_size.cy);\t\t//设置最小高度\n\n    CDialog::OnGetMinMaxInfo(lpMMI);\n}\n\n\nvoid CBaseDialog::OnSize(UINT nType, int cx, int cy)\n{\n    CDialog::OnSize(nType, cx, cy);\n\n    // TODO: 在此处添加消息处理程序代码\n    if (nType != SIZE_MAXIMIZED && nType != SIZE_MINIMIZED)\n    {\n        //m_window_width = cx;\n        //m_window_hight = cy;\n        CRect rect;\n        GetWindowRect(&rect);\n        m_window_size.cx = rect.Width();\n        m_window_size.cy = rect.Height();\n    }\n\n}\n\n\nINT_PTR CBaseDialog::DoModal()\n{\n    HWND unique_hwnd{ m_unique_hwnd[GetDialogName()] };\n    if (unique_hwnd != NULL && !GetDialogName().IsEmpty())      ///如果对话框已存在，则显示已存在的对话框\n    {\n        ::ShowWindow(unique_hwnd, SW_RESTORE);\n        ::SetForegroundWindow(unique_hwnd);\n        return 0;\n    }\n    return CDialog::DoModal();\n}\n"
  },
  {
    "path": "TrafficMonitor/BaseDialog.h",
    "content": "﻿#pragma once\n\n\n// CBaseDialog 对话框\n//用于实现记住对话框大小\n//并将窗口初始大小设置为最小大小\n\nclass CBaseDialog : public CDialog\n{\n    DECLARE_DYNAMIC(CBaseDialog)\n\npublic:\n    CBaseDialog(UINT nIDTemplate, CWnd* pParent = NULL);   // 标准构造函数\n    virtual ~CBaseDialog();\n\n    // 对话框数据\n    //#ifdef AFX_DESIGN_TIME\n    //\tenum { IDD = IDD_BASEDIALOG };\n    //#endif\n\n    void SetMinSize(int cx, int cy);\t\t//设置窗口的最小大小，如果未设置，则使用窗口的初始大小作为最小大小\n    static HWND GetUniqueHandel(LPCTSTR dlg_name);          //获指定窗口唯一的句柄\n    static const std::map<CString, HWND>& AllUniqueHandels();   //获取所有窗口的句柄\n\nprivate:\n    void LoadConfig();\n    void SaveConfig() const;\n\nprivate:\n    CSize m_min_size{};     //窗口的最小大小（以 96dpi 的大小保存）\n    CSize m_window_size{ -1, -1 };\n    static std::map<CString, HWND> m_unique_hwnd;        //针对每一个基类的唯一的窗口句柄\n\nprotected:\n    virtual CString GetDialogName() const = 0;\n    void EnableDlgCtrl(UINT id, bool enable);\n\n    virtual void DoDataExchange(CDataExchange* pDX);    // DDX/DDV 支持\n\n    DECLARE_MESSAGE_MAP()\npublic:\n    virtual BOOL OnInitDialog();\n    afx_msg void OnDestroy();\n    afx_msg void OnGetMinMaxInfo(MINMAXINFO* lpMMI);\n    afx_msg void OnSize(UINT nType, int cx, int cy);\n    virtual INT_PTR DoModal();\n};\n"
  },
  {
    "path": "TrafficMonitor/CAutoAdaptSettingsDlg.cpp",
    "content": "﻿// CAutoAdaptSettingsDlg.cpp: 实现文件\n//\n\n#include \"stdafx.h\"\n#include \"TrafficMonitor.h\"\n#include \"CAutoAdaptSettingsDlg.h\"\n#include \"afxdialogex.h\"\n\n\n// CAutoAdaptSettingsDlg 对话框\n\nIMPLEMENT_DYNAMIC(CAutoAdaptSettingsDlg, CDialog)\n\nCAutoAdaptSettingsDlg::CAutoAdaptSettingsDlg(TaskBarSettingData& data, CWnd* pParent /*=nullptr*/)\n\t: CDialog(IDD_ATUO_ADAPT_SETTING_DIALOG, pParent), m_data(data)\n{\n\n}\n\nCAutoAdaptSettingsDlg::~CAutoAdaptSettingsDlg()\n{\n}\n\nvoid CAutoAdaptSettingsDlg::InitComboBox(CComboBox& combo_box, int style_sel)\n{\n\tfor (int i = 0; i < TASKBAR_DEFAULT_STYLE_NUM; i++)\n\t\tcombo_box.AddString(CCommon::LoadText(IDS_PRESET, std::to_wstring(i + 1).c_str()));\n\tif (style_sel >= 0 && style_sel < TASKBAR_DEFAULT_STYLE_NUM)\n\t\tcombo_box.SetCurSel(style_sel);\n\telse\n\t\tcombo_box.SetCurSel(0);\n}\n\nint CAutoAdaptSettingsDlg::GetComboBoxSel(const CComboBox& combo_box)\n{\n\tint sel = combo_box.GetCurSel();\n\tif (sel >= 0 && sel < TASKBAR_DEFAULT_STYLE_NUM)\n\t\treturn sel;\n\telse\n\t\treturn 0;\n}\n\nvoid CAutoAdaptSettingsDlg::DoDataExchange(CDataExchange* pDX)\n{\n\tCDialog::DoDataExchange(pDX);\n\tDDX_Control(pDX, IDC_DARK_MODE_DEFAULT_STYLE_COMBO, m_dark_mode_default_style_combo);\n\tDDX_Control(pDX, IDC_LIGHT_MODE_DEFAULT_STYLE_COMBO, m_light_mode_default_style_combo);\n}\n\n\nBEGIN_MESSAGE_MAP(CAutoAdaptSettingsDlg, CDialog)\nEND_MESSAGE_MAP()\n\n\n// CAutoAdaptSettingsDlg 消息处理程序\n\n\nBOOL CAutoAdaptSettingsDlg::OnInitDialog()\n{\n\tCDialog::OnInitDialog();\n\n\t// TODO:  在此添加额外的初始化\n\tInitComboBox(m_dark_mode_default_style_combo, m_data.dark_default_style);\n\tInitComboBox(m_light_mode_default_style_combo, m_data.light_default_style);\n    CheckDlgButton(IDC_AUTO_SAVE_TO_PRESET_CHECK, m_data.auto_save_taskbar_color_settings_to_preset);\n\n    m_toolTip.Create(this);\n    m_toolTip.SetMaxTipWidth(theApp.DPI(300));\n    m_toolTip.AddTool(GetDlgItem(IDC_AUTO_SAVE_TO_PRESET_CHECK), CCommon::LoadText(IDS_AUTO_SAVE_TO_PRESET_TIP));\n\n\treturn TRUE;  // return TRUE unless you set the focus to a control\n\t\t\t\t  // 异常: OCX 属性页应返回 FALSE\n}\n\n\nvoid CAutoAdaptSettingsDlg::OnOK()\n{\n\t// TODO: 在此添加专用代码和/或调用基类\n\n\t//获取控件中的设置\n\tm_data.dark_default_style = GetComboBoxSel(m_dark_mode_default_style_combo);\n\tm_data.light_default_style = GetComboBoxSel(m_light_mode_default_style_combo);\n    m_data.auto_save_taskbar_color_settings_to_preset = (IsDlgButtonChecked(IDC_AUTO_SAVE_TO_PRESET_CHECK) != 0);\n\n\tCDialog::OnOK();\n}\n\n\nBOOL CAutoAdaptSettingsDlg::PreTranslateMessage(MSG* pMsg)\n{\n    // TODO: 在此添加专用代码和/或调用基类\n    if (pMsg->message == WM_MOUSEMOVE)\n        m_toolTip.RelayEvent(pMsg);\n\n    return CDialog::PreTranslateMessage(pMsg);\n}\n"
  },
  {
    "path": "TrafficMonitor/CAutoAdaptSettingsDlg.h",
    "content": "﻿#pragma once\n\n\n// CAutoAdaptSettingsDlg 对话框\n\nclass CAutoAdaptSettingsDlg : public CDialog\n{\n\tDECLARE_DYNAMIC(CAutoAdaptSettingsDlg)\n\npublic:\n\tCAutoAdaptSettingsDlg(TaskBarSettingData& data, CWnd* pParent = nullptr);   // 标准构造函数\n\tvirtual ~CAutoAdaptSettingsDlg();\n\n// 对话框数据\n#ifdef AFX_DESIGN_TIME\n\tenum { IDD = IDD_ATUO_ADAPT_SETTING_DIALOG };\n#endif\n\nprivate:\n\tCComboBox m_dark_mode_default_style_combo;\n\tCComboBox m_light_mode_default_style_combo;\n\tTaskBarSettingData& m_data;\n    CToolTipCtrl m_toolTip;\n\nprivate:\n\tvoid InitComboBox(CComboBox& combo_box, int style_sel);\n\tint GetComboBoxSel(const CComboBox& combo_box);\n\nprotected:\n\tvirtual void DoDataExchange(CDataExchange* pDX);    // DDX/DDV 支持\n\n\tDECLARE_MESSAGE_MAP()\npublic:\n\tvirtual BOOL OnInitDialog();\n\tvirtual void OnOK();\n    virtual BOOL PreTranslateMessage(MSG* pMsg);\n};\n"
  },
  {
    "path": "TrafficMonitor/CMFCColorDialogEx.cpp",
    "content": "﻿// CMFCColorDialogEx.cpp: 实现文件\n//\n\n#include \"stdafx.h\"\n#include \"CMFCColorDialogEx.h\"\n#include \"Common.h\"\n\n// CMFCColorDialogEx\n\n//IMPLEMENT_DYNAMIC(CMFCColorDialogEx, CMFCColorDialog)\n\nCMFCColorDialogEx::CMFCColorDialogEx(COLORREF clrInit, DWORD dwFlags, CWnd* pParentWnd, HPALETTE hPal)\n\t:CMFCColorDialog(clrInit, dwFlags, pParentWnd, hPal)\n{\n\n}\n\nCMFCColorDialogEx::~CMFCColorDialogEx()\n{\n}\n\n\nBEGIN_MESSAGE_MAP(CMFCColorDialogEx, CMFCColorDialog)\nEND_MESSAGE_MAP()\n\n\n\n// CMFCColorDialogEx 消息处理程序\n\nBOOL CMFCColorDialogEx::OnInitDialog()\n{\n\tCMFCColorDialog::OnInitDialog();\n\n\t// TODO:  在此添加额外的初始化\n\n\t//设置控件字体\n\tCWnd* pParent = GetParent();\n\tif (pParent != nullptr)\n\t{\n\t\tCCommon::SetDialogFont(this, pParent->GetFont());\n\n\t\tCWnd* pPropSheet = m_pColourSheetOne->GetParent();\n\t\tif (pPropSheet != nullptr)\n\t\t\tCCommon::SetDialogFont(pPropSheet, pParent->GetFont());\n\n\t\t//CCommon::SetDialogFont(m_pColourSheetOne, pParent->GetFont());\n\t\t//CCommon::SetDialogFont(m_pColourSheetTwo, pParent->GetFont());\n\t}\n\n\n\treturn TRUE;  // return TRUE unless you set the focus to a control\n\t\t\t\t  // 异常: OCX 属性页应返回 FALSE\n}\n\n"
  },
  {
    "path": "TrafficMonitor/CMFCColorDialogEx.h",
    "content": "#pragma once\n\n\n// CMFCColorDialogEx\n\nclass CMFCColorDialogEx : public CMFCColorDialog\n{\n\t//DECLARE_DYNAMIC(CMFCColorDialogEx)\n\npublic:\n\tCMFCColorDialogEx(COLORREF clrInit = 0, DWORD dwFlags = 0 /* reserved */, CWnd* pParentWnd = NULL, HPALETTE hPal = NULL);\n\tvirtual ~CMFCColorDialogEx();\n\nprotected:\n\tDECLARE_MESSAGE_MAP()\n\n\tvirtual BOOL OnInitDialog();\n\n};\n\n\n"
  },
  {
    "path": "TrafficMonitor/CPUUsage.cpp",
    "content": "#include \"stdafx.h\"\n#include \"CPUUsage.h\"\n#include \"Common.h\"\n#include \"TrafficMonitor.h\"\n\nvoid CCPUUsage::SetUseCPUTimes(bool use_get_system_times)\n{\n\tif (m_use_get_system_times != use_get_system_times)\n\t{\n\t\tm_use_get_system_times = use_get_system_times;\n\t\tm_first_get_CPU_utility = true;\n\t}\n}\n\nint CCPUUsage::GetCPUUsage()\n{\n\tif (m_use_get_system_times)\n\t\treturn GetCPUUsageByGetSystemTimes();\n\telse\n\t\treturn GetCPUUsageByPdh();\n}\n\nint CCPUUsage::GetCPUUsageByGetSystemTimes()\n{\n\tint cpu_usage{};\n\tFILETIME idleTime;\n\tFILETIME kernelTime;\n\tFILETIME userTime;\n\tGetSystemTimes(&idleTime, &kernelTime, &userTime);\n\n\t__int64 idle = CCommon::CompareFileTime2(m_preidleTime, idleTime);\n\t__int64 kernel = CCommon::CompareFileTime2(m_prekernelTime, kernelTime);\n\t__int64 user = CCommon::CompareFileTime2(m_preuserTime, userTime);\n\n\tif (kernel + user == 0)\n\t{\n\t\tcpu_usage = 0;\n\t}\n\telse\n\t{\n\t\t//ܵʱ-ʱ䣩/ܵʱ=ռcpuʱʹ\n\t\tcpu_usage = static_cast<int>(abs((kernel + user - idle) * 100 / (kernel + user)));\n\t}\n\tm_preidleTime = idleTime;\n\tm_prekernelTime = kernelTime;\n\tm_preuserTime = userTime;\n\n\treturn cpu_usage;\n}\n\nint CCPUUsage::GetCPUUsageByPdh()\n{\n\tint cpu_usage{};\n\tHQUERY hQuery;\n\tHCOUNTER hCounter;\n\tDWORD counterType;\n\tPDH_RAW_COUNTER rawData;\n\n\tPdhOpenQuery(NULL, 0, &hQuery);//ʼѯ\n\tconst wchar_t* query_str{};\n\tif (theApp.m_win_version.GetMajorVersion() >= 10)\n\t\tquery_str = L\"\\\\Processor Information(_Total)\\\\% Processor Utility\";\n\telse\n\t\tquery_str = L\"\\\\Processor Information(_Total)\\\\% Processor Time\";\n\tPdhAddCounter(hQuery, query_str, NULL, &hCounter);\n\tPdhCollectQueryData(hQuery);\n\tPdhGetRawCounterValue(hCounter, &counterType, &rawData);\n\n\tif (m_first_get_CPU_utility) {//ҪݲܼCPUʹ\n\t\tcpu_usage = 0;\n\t\tm_first_get_CPU_utility = false;\n\t}\n\telse {\n\t\tPDH_FMT_COUNTERVALUE fmtValue;\n\t\tPdhCalculateCounterFromRawValue(hCounter, PDH_FMT_DOUBLE, &rawData, &m_last_rawData, &fmtValue);//ʹ\n\t\tcpu_usage = fmtValue.doubleValue;//\n\t\tif (cpu_usage > 100)\n\t\t\tcpu_usage = 100;\n\t}\n\tm_last_rawData = rawData;//һ\n\tPdhCloseQuery(hQuery);//رղѯ\n\treturn cpu_usage;\n}\n"
  },
  {
    "path": "TrafficMonitor/CPUUsage.h",
    "content": "#pragma once\n#include <Pdh.h>\n#include <PdhMsg.h>\n#pragma comment(lib,\"pdh.lib\")\n\nclass CCPUUsage\n{\npublic:\n\tCCPUUsage()\n\t{}\n\n\t~CCPUUsage()\n\t{}\n\n\tvoid SetUseCPUTimes(bool use_get_system_times);\t\t//ûȡCPUʵķʽͨGetSystemTimesPdh\n\tint GetCPUUsage();\n\nprivate:\n\tint GetCPUUsageByGetSystemTimes();\n\tint GetCPUUsageByPdh();\n\nprivate:\n\tbool m_use_get_system_times{ true };\t\t//ǷʹGetSysTimeAPIȡCPU\n\n\tPDH_RAW_COUNTER m_last_rawData;//CPUʹʵһ\n\tbool m_first_get_CPU_utility{ true };\n\n\tFILETIME m_preidleTime{};\n\tFILETIME m_prekernelTime{};\n\tFILETIME m_preuserTime{};\n\n};\n\n"
  },
  {
    "path": "TrafficMonitor/CSkinPreviewView.cpp",
    "content": "﻿// CSkinPreviewView.cpp: 实现文件\n//\n\n#include \"stdafx.h\"\n#include \"TrafficMonitor.h\"\n#include \"CSkinPreviewView.h\"\n\n\n// CSkinPreviewView\n\nIMPLEMENT_DYNCREATE(CSkinPreviewView, CScrollView)\n\nCSkinPreviewView::CSkinPreviewView()\n{\n\n}\n\nCSkinPreviewView::~CSkinPreviewView()\n{\n}\n\n\nBEGIN_MESSAGE_MAP(CSkinPreviewView, CScrollView)\nEND_MESSAGE_MAP()\n\n\n// CSkinPreviewView 绘图\n\nvoid CSkinPreviewView::OnInitialUpdate()\n{\n\tCScrollView::OnInitialUpdate();\n\n\tCSize sizeTotal;\n\t// TODO:  计算此视图的合计大小\n\tm_size.cx = 0;\n\tm_size.cy = 0;\n\tSetScrollSizes(MM_TEXT, m_size);\n\n\n}\n\nvoid CSkinPreviewView::OnDraw(CDC* pDC)\n{\n\tCDocument* pDoc = GetDocument();\n\t// TODO:  在此添加绘制代码\n    CRect draw_rect(CPoint(0, 0), m_size);\n    m_skin_data->DrawPreview(pDC, draw_rect);\n}\n\n\n// CSkinPreviewView 诊断\n\n//#ifdef _DEBUG\n//void CSkinPreviewView::AssertValid() const\n//{\n//\tCScrollView::AssertValid();\n//}\n\n//#ifndef _WIN32_WCE\n//void CSkinPreviewView::Dump(CDumpContext& dc) const\n//{\n//\tCScrollView::Dump(dc);\n//}\n//#endif\n//#endif //_DEBUG\n\nvoid CSkinPreviewView::InitialUpdate()\n{\n\tOnInitialUpdate();\n}\nvoid CSkinPreviewView::SetSize(int width, int hight)\n{\n\tm_size = CSize(width, hight);\n\tSetScrollSizes(MM_TEXT, m_size);\n}\n\n// CSkinPreviewView 消息处理程序\n"
  },
  {
    "path": "TrafficMonitor/CSkinPreviewView.h",
    "content": "﻿#pragma once\n#include \"DrawCommon.h\"\n#include \"SkinFile.h\"\n\n\n// CSkinPreviewView 视图\n\n\nclass CSkinPreviewView : public CScrollView\n{\n\tDECLARE_DYNCREATE(CSkinPreviewView)\n\nprotected:\n\tCSkinPreviewView();           // 动态创建所使用的受保护的构造函数\n\tvirtual ~CSkinPreviewView();\n\npublic:\n//#ifdef _DEBUG\n//\tvirtual void AssertValid() const;\n//#ifndef _WIN32_WCE\n//\tvirtual void Dump(CDumpContext& dc) const;\n//#endif\n//#endif\n\n//成员函数\npublic:\n\tvoid InitialUpdate();\n\tvoid SetSize(int width, int hight);\n\tvoid SetSkinData(CSkinFile* skin_data) { m_skin_data = skin_data; }\n\n\t//成员变量\nprotected:\n\tCSize m_size;\n\tCPoint m_start_point;\t\t\t//绘图的起始位置\n\n    CSkinFile* m_skin_data;\n\nprotected:\n\tvirtual void OnDraw(CDC* pDC);      // 重写以绘制该视图\n\tvirtual void OnInitialUpdate();     // 构造后的第一次\n\n\tDECLARE_MESSAGE_MAP()\n};\n\n\n"
  },
  {
    "path": "TrafficMonitor/CTabCtrlEx.cpp",
    "content": "﻿// CTabCtrlEx.cpp: 实现文件\n//\n\n#include \"stdafx.h\"\n#include \"CTabCtrlEx.h\"\n\n\n// CTabCtrlEx\n\nIMPLEMENT_DYNAMIC(CTabCtrlEx, CTabCtrl)\n\nCTabCtrlEx::CTabCtrlEx()\n{\n\n}\n\nCTabCtrlEx::~CTabCtrlEx()\n{\n}\n\nvoid CTabCtrlEx::AddWindow(CWnd* pWnd, LPCTSTR lable_text)\n{\n\tif (pWnd == nullptr || pWnd->GetSafeHwnd() == NULL)\n\t\treturn;\n\n    InsertItem(m_tab_list.size(), lable_text, m_tab_list.size());\n\n\tpWnd->SetParent(this);\n\tpWnd->MoveWindow(m_tab_rect);\n\n\tm_tab_list.push_back(pWnd);\n}\n\nvoid CTabCtrlEx::SetCurTab(int index)\n{\n    if (index < 0 || index >= static_cast<int>(m_tab_list.size()))\n        index = 0;\n\tSetCurSel(index);\n\n\tint tab_size = m_tab_list.size();\n\tfor (int i = 0; i < tab_size; i++)\n\t{\n\t\tif (i == index)\n\t\t{\n\t\t\tm_tab_list[i]->ShowWindow(SW_SHOW);\n\t\t\tm_tab_list[i]->SetFocus();\n\t\t}\n\t\telse\n\t\t{\n\t\t\tm_tab_list[i]->ShowWindow(SW_HIDE);\n\t\t}\n\t}\n}\n\nCWnd* CTabCtrlEx::GetCurrentTab()\n{\n    int cur_tab_index = GetCurSel();\n    if (cur_tab_index >= 0 && cur_tab_index < m_tab_list.size())\n    {\n        return m_tab_list[cur_tab_index];\n    }\n    return nullptr;\n}\n\nvoid CTabCtrlEx::AdjustTabWindowSize()\n{\n    CalSubWindowSize();\n    for (size_t i{}; i < m_tab_list.size(); i++)\n    {\n        m_tab_list[i]->MoveWindow(m_tab_rect);\n    }\n}\n\nvoid CTabCtrlEx::CalSubWindowSize()\n{\n    GetClientRect(m_tab_rect);\n    CRect rc_temp = m_tab_rect;\n    AdjustRect(FALSE, rc_temp);\n    int margin = rc_temp.left - m_tab_rect.left;\n    CRect rcTabItem;\n    GetItemRect(0, rcTabItem);\n    m_tab_rect.top += rcTabItem.Height() + margin;\n    m_tab_rect.left += margin;\n    m_tab_rect.bottom -= margin;\n    m_tab_rect.right -= margin;\n}\n\n\nBEGIN_MESSAGE_MAP(CTabCtrlEx, CTabCtrl)\n\tON_NOTIFY_REFLECT(TCN_SELCHANGE, &CTabCtrlEx::OnTcnSelchange)\n    ON_WM_SIZE()\nEND_MESSAGE_MAP()\n\n\n\n// CTabCtrlEx 消息处理程序\n\n\n\n\nvoid CTabCtrlEx::OnTcnSelchange(NMHDR *pNMHDR, LRESULT *pResult)\n{\n\t// TODO: 在此添加控件通知处理程序代码\n\tint tab_selected = GetCurSel();\n\tSetCurTab(tab_selected);\n\n\t*pResult = 0;\n}\n\n\nvoid CTabCtrlEx::PreSubclassWindow()\n{\n\t// TODO: 在此添加专用代码和/或调用基类\n\n\t//计算子窗口的位置\n    CalSubWindowSize();\n\n\tCTabCtrl::PreSubclassWindow();\n}\n\n\nvoid CTabCtrlEx::OnSize(UINT nType, int cx, int cy)\n{\n    CTabCtrl::OnSize(nType, cx, cy);\n\n    // TODO: 在此处添加消息处理程序代码\n    AdjustTabWindowSize();\n}\n"
  },
  {
    "path": "TrafficMonitor/CTabCtrlEx.h",
    "content": "#pragma once\n\n\n// CTabCtrlEx\n\nclass CTabCtrlEx : public CTabCtrl\n{\n\tDECLARE_DYNAMIC(CTabCtrlEx)\n\npublic:\n\tCTabCtrlEx();\n\tvirtual ~CTabCtrlEx();\n\n\tvoid AddWindow(CWnd* pWnd, LPCTSTR lable_text);\t\t//ǰtabؼһӴ\n\tvoid SetCurTab(int index);\n    CWnd* GetCurrentTab();\n    void AdjustTabWindowSize();\n\nprotected:\n    void CalSubWindowSize();\n\n\tDECLARE_MESSAGE_MAP()\n\nprotected:\n\tvector<CWnd*> m_tab_list;\t\t//tabؼÿӴڵָ\npublic:\n\tafx_msg void OnTcnSelchange(NMHDR *pNMHDR, LRESULT *pResult);\n\tvirtual void PreSubclassWindow();\n\n\tCRect m_tab_rect;\n    afx_msg void OnSize(UINT nType, int cx, int cy);\n};\n\n\n"
  },
  {
    "path": "TrafficMonitor/CVariant.cpp",
    "content": "#include \"stdafx.h\"\n#include \"CVariant.h\"\n\n\nCVariant::CVariant(int value)\n{\n\tm_value_int = value;\n\tm_type = eType::INT;\n}\n\nCVariant::CVariant(size_t value)\n{\n\tm_value_int = static_cast<int>(value);\n\tm_type = eType::UINT;\n}\n\nCVariant::CVariant(double value)\n{\n\tm_value_double = value;\n\tm_type = eType::DOUBLE;\n}\n\nCVariant::CVariant(LPCTSTR value)\n{\n\tm_value_string = value;\n\tm_type = eType::STRING;\n}\n\nCVariant::CVariant(const CString& value)\n{\n\tm_value_string = value;\n\tm_type = eType::STRING;\n}\n\nCVariant::CVariant(const wstring & value)\n{\n\tm_value_string = value.c_str();\n\tm_type = eType::STRING;\n}\n\nCVariant::~CVariant()\n{\n}\n\nCString CVariant::ToString() const\n{\n\tCString str;\n\tswitch (m_type)\n\t{\n\tcase CVariant::eType::INT:\n\t\tstr.Format(_T(\"%d\"), m_value_int);\n\t\tbreak;\n\tcase eType::UINT:\n\t\tstr.Format(_T(\"%u\"), static_cast<unsigned int>(m_value_int));\n\t\tbreak;\n\tcase CVariant::eType::DOUBLE:\n\t\tstr.Format(_T(\"%g\"), m_value_double);\n\t\tbreak;\n\tcase CVariant::eType::STRING:\n\t\tstr = m_value_string;\n\t\tbreak;\n\tdefault:\n\t\tbreak;\n\t}\n\treturn str;\n}\n"
  },
  {
    "path": "TrafficMonitor/CVariant.h",
    "content": "#pragma once\nclass CVariant\n{\npublic:\n\tCVariant(int value);\n\tCVariant(size_t value);\n\tCVariant(double value);\n\tCVariant(LPCTSTR value);\n\tCVariant(const CString& value);\n\tCVariant(const wstring& value);\n\n\t~CVariant();\n\n\tCString ToString() const;\n\nprivate:\n\tenum class eType { INT, UINT, DOUBLE, STRING };\n\n\tint m_value_int;\n\tdouble m_value_double;\n\tCString m_value_string;\n\teType m_type;\n\n};\n\n"
  },
  {
    "path": "TrafficMonitor/CalendarHelper.cpp",
    "content": "#include \"stdafx.h\"\n#include \"CalendarHelper.h\"\n\n\nCCalendarHelper::CCalendarHelper()\n{\n}\n\n\nCCalendarHelper::~CCalendarHelper()\n{\n}\n\nbool CCalendarHelper::IsLeapYear(int year)\n{\n\treturn ((0 == year % 4 && 0 != year % 100) || 0 == year % 400);\n}\n\nint CCalendarHelper::CaculateWeekDay(int y, int m, int d)\n{\n\tif (m <= 2)\n\t{\n\t\tm += 12;\n\t\ty--;\n\t}\n\treturn (d + 2 * m + 3 * (m + 1) / 5 + y + y / 4 - y / 100 + y / 400 + 1) % 7;\n}\n\nint CCalendarHelper::DaysInMonth(int year, int month)\n{\n\tbool leap{ IsLeapYear(year) };\n\tswitch (month)\n\t{\n\tcase 2:\n\t\tif (leap) return 29;\n\t\telse return 28;\n\tcase 4: case 6: case 9: case 11:\n\t\treturn 30;\n\tdefault:\n\t\treturn 31;\n\t}\n}\n\nvoid CCalendarHelper::GetCalendar(int year, int month, DayTraffic calendar[CALENDAR_HEIGHT][CALENDAR_WIDTH], bool sunday_first)\n{\n\tmemset(calendar, 0, sizeof(DayTraffic)*CALENDAR_HEIGHT*CALENDAR_WIDTH);\n\tint days{ DaysInMonth(year, month) };\n\tint first_weak_day{ CaculateWeekDay(year, month, 1) };\n\tif(!sunday_first)\n\t{\n\t\tfirst_weak_day--;\n\t\tif (first_weak_day < 0)\n\t\t\tfirst_weak_day = 6;\n\t}\n\tint i{}, j{};\n\tfor (int n{}; n < 37; n++)\n\t{\n\t\tif (n < first_weak_day)\n\t\t{\n\t\t\tcalendar[i][j].day = 0;\n\t\t}\n\t\telse\n\t\t{\n\t\t\tint day = n - first_weak_day + 1;\n\t\t\tif (day <= days)\n\t\t\t\tcalendar[i][j].day = day;\n\t\t}\n\t\tj++;\n\t\tif (j >= 7)\n\t\t{\n\t\t\tj = 0;\n\t\t\ti++;\n\t\t}\n\t}\n}\n"
  },
  {
    "path": "TrafficMonitor/CalendarHelper.h",
    "content": "#pragma once\n#define CALENDAR_WIDTH 7\n#define CALENDAR_HEIGHT 6\n\nstruct DayTraffic\n{\n\tint day;\n\t__int64 up_traffic;\n\t__int64 down_traffic;\n\tbool mixed;\n\tCRect rect;\n\n\t__int64 traffic() const\n\t{\n\t\treturn up_traffic + down_traffic;\n\t}\n};\n\nclass CCalendarHelper\n{\npublic:\n\tCCalendarHelper();\n\t~CCalendarHelper();\n\n\t//ǷΪ\n\tstatic bool IsLeapYear(int year);\n\t//ڼ(0~6~)\n\tstatic int CaculateWeekDay(int y, int m, int d);\n\t//һж\n\tstatic int DaysInMonth(int year, int month);\n\n\t//ȡָ·ݵݣcalendar\n\t//sunday_firstΪtrueΪÿܵĵһ죬򣬽һΪÿܵĵһ\n\tstatic void GetCalendar(int year, int month, DayTraffic calendar[CALENDAR_HEIGHT][CALENDAR_WIDTH], bool sunday_first = true);\n\n};\n\n"
  },
  {
    "path": "TrafficMonitor/ColorSettingListCtrl.cpp",
    "content": "﻿#include \"stdafx.h\"\n#include \"ColorSettingListCtrl.h\"\n\nIMPLEMENT_DYNAMIC(CColorSettingListCtrl, CListCtrl)\n\nCColorSettingListCtrl::CColorSettingListCtrl()\n{\n}\n\n\nCColorSettingListCtrl::~CColorSettingListCtrl()\n{\n}\n\nvoid CColorSettingListCtrl::SetItemColor(int row, int col, COLORREF color)\n{\n    m_colors[row][col] = color;\n}\n\n\nCOLORREF CColorSettingListCtrl::GetItemColor(int row, int col)\n{\n    return m_colors[row][col];\n}\n\nBEGIN_MESSAGE_MAP(CColorSettingListCtrl, CListCtrl)\n\tON_NOTIFY_REFLECT(NM_CUSTOMDRAW, &CColorSettingListCtrl::OnNMCustomdraw)\nEND_MESSAGE_MAP()\n\n\nvoid CColorSettingListCtrl::OnNMCustomdraw(NMHDR *pNMHDR, LRESULT *pResult)\n{\n\t*pResult = CDRF_DODEFAULT;\n\tLPNMLVCUSTOMDRAW lplvdr = reinterpret_cast<LPNMLVCUSTOMDRAW>(pNMHDR);\n\tNMCUSTOMDRAW& nmcd = lplvdr->nmcd;\n\tswitch (lplvdr->nmcd.dwDrawStage)\t//判断状态   \n\t{\n\tcase CDDS_PREPAINT:\n\t\t*pResult = CDRF_NOTIFYITEMDRAW;\n\t\tbreak;\n\tcase CDDS_ITEMPREPAINT:\t\t\t//如果为画ITEM之前就要进行颜色的改变\n\t\tif (nmcd.dwItemSpec >= 0 && nmcd.dwItemSpec < GetItemCount())\n\t\t{\n\t\t\t//double range = m_item_rage_data[nmcd.dwItemSpec].data_value;\n\t\t\t//CDC* pDC = CDC::FromHandle(nmcd.hdc);\t\t//获取绘图DC\n\t\t\t//CRect item_rect, draw_rect;\n\t\t\t//GetSubItemRect(nmcd.dwItemSpec,m_draw_item_range_row, LVIR_BOUNDS, item_rect);\t//获取绘图单元格的矩形区域\n\t\t\t//CDrawCommon::SetDrawRect(pDC, item_rect);\t\t//设置绘图区域为当前列\n\t\t\t//draw_rect = item_rect;\n\t\t\t//if (draw_rect.Height() > 2 * m_margin)\n\t\t\t//{\n\t\t\t//\tdraw_rect.top += m_margin;\n\t\t\t//\tdraw_rect.bottom -= m_margin;\n\t\t\t//}\n\t\t\t//int width;\n\t\t\t//if (m_use_log_scale)\t//使用对数比例（y=ln(x+1)）\n\t\t\t//{\n\t\t\t//\trange = std::log(range + 1);\n\t\t\t//\twidth = static_cast<int>(range*draw_rect.Width() / std::log(1000 + 1));\n\t\t\t//}\n\t\t\t//else\t\t//使用线性比例（y=x）\n\t\t\t//{\n\t\t\t//\twidth = static_cast<int>(range*draw_rect.Width() / 1000);\n\t\t\t//}\n\t\t\t//draw_rect.right = draw_rect.left + width;\n\t\t\t//pDC->FillSolidRect(draw_rect, m_item_rage_data[nmcd.dwItemSpec].color);\n\n\t\t\t////当前列绘制完成后将绘图区域设置为左边的区域，防止当前列的区域被覆盖\n\t\t\t//CRect rect1{ item_rect };\n\t\t\t//rect1.left = 0;\n\t\t\t//rect1.right = item_rect.left;\n\t\t\t//CDrawCommon::SetDrawRect(pDC, rect1);\n\n            CDC* pDC = CDC::FromHandle(nmcd.hdc);\t\t//获取绘图DC\n            CRect item_rect;\n            auto& col_color_map = m_colors[nmcd.dwItemSpec];\n            for (auto iter = col_color_map.begin(); iter != col_color_map.end(); ++iter)\n            {\n                GetSubItemRect(nmcd.dwItemSpec, iter->first, LVIR_BOUNDS, item_rect);\t//获取绘图单元格的矩形区域\n                //设置绘图剪辑区域\n                CDrawCommon::SetDrawRect(pDC, item_rect);\n                //使用双缓冲绘图\n                CDrawDoubleBuffer draw_double_buffer(pDC, item_rect);\n                //填充背景\n                item_rect.MoveToXY(0, 0);\n                draw_double_buffer.GetMemDC()->FillSolidRect(item_rect, GetSysColor(COLOR_WINDOW));\n                //绘制颜色矩形\n                item_rect.DeflateRect(m_margin, m_margin);\n                draw_double_buffer.GetMemDC()->FillSolidRect(item_rect, iter->second);\n                //绘制矩形边框\n                CDrawCommon drawer;\n                drawer.Create(draw_double_buffer.GetMemDC(), this);\n                drawer.DrawRectOutLine(item_rect, RGB(192, 192, 192));\n            }\n            //当前列绘制完成后将绘图区域设置为第一列的区域，防止颜色列的区域被覆盖\n            CRect rect1{};\n            GetSubItemRect(nmcd.dwItemSpec, 1, LVIR_BOUNDS, rect1);\t//获取第1列单元格的矩形区域\n            rect1.right = rect1.left;\n            rect1.left = 0;\n            CDrawCommon::SetDrawRect(pDC, rect1);\n\n\t\t}\n\t\t*pResult = CDRF_DODEFAULT;\n\t\tbreak;\n\t}\n}\n"
  },
  {
    "path": "TrafficMonitor/ColorSettingListCtrl.h",
    "content": "#pragma once\n#include \"afxcmn.h\"\n#include \"DrawCommon.h\"\nclass CColorSettingListCtrl :\n\tpublic CListCtrl\n{\nDECLARE_DYNAMIC(CColorSettingListCtrl)\npublic:\n\tCColorSettingListCtrl();\n\t~CColorSettingListCtrl();\n\n    void SetItemColor(int row, int col, COLORREF color);\n    COLORREF GetItemColor(int row, int col);\n\tvoid SetDrawItemRangMargin(int margin) { m_margin = margin; }\t\t\t//ûƻƵľεıԵԪ߿ľΣֵԽƵľԽϸǲܳбоһ\n\nprotected:\n\tint m_margin{};\n\tstd::map<int, std::map<int, COLORREF>> m_colors;\t//ڱÿһÿһеɫ\n\n\tDECLARE_MESSAGE_MAP()\n\tafx_msg void OnNMCustomdraw(NMHDR *pNMHDR, LRESULT *pResult);\n};\n\n"
  },
  {
    "path": "TrafficMonitor/ColorStatic.cpp",
    "content": "﻿// ColorStatic.cpp : 实现文件\n//\n\n#include \"stdafx.h\"\n#include \"TrafficMonitor.h\"\n#include \"ColorStatic.h\"\n#include \"DrawCommon.h\"\n\n// CColorStatic\n\nIMPLEMENT_DYNAMIC(CColorStatic, CStatic)\n\nCColorStatic::CColorStatic()\n{\n\n}\n\nCColorStatic::~CColorStatic()\n{\n}\n\nvoid CColorStatic::SetFillColor(COLORREF fill_color)\n{\n\tm_colors.resize(1);\n\tm_colors[0] = fill_color;\n\t//m_fill_color = fill_color;\n\tInvalidate();\n}\n\nvoid CColorStatic::SetColorNum(int color_num)\n{\n\tif (color_num <= 0 || color_num > 32)\n\t\tcolor_num = 1;\n\tm_colors.resize(color_num);\n}\n\nvoid CColorStatic::SetFillColor(int index, COLORREF fill_color)\n{\n\tif (index >= 0 && index < static_cast<int>(m_colors.size()))\n\t\tm_colors[index] = fill_color;\n}\n\nvoid CColorStatic::SetLinkCursor(bool link_cursor)\n{\n\tm_link_cursor = link_cursor;\n}\n\nvoid CColorStatic::EnableWindow(bool enable)\n{\n    CStatic::EnableWindow(enable);\n    Invalidate(FALSE);\n}\n\n\nBEGIN_MESSAGE_MAP(CColorStatic, CStatic)\n\tON_WM_PAINT()\n\tON_WM_MOUSEHOVER()\n\tON_WM_MOUSELEAVE()\n\tON_WM_SETCURSOR()\n\tON_WM_MOUSEMOVE()\n\tON_WM_LBUTTONUP()\nEND_MESSAGE_MAP()\n\n\n\n// CColorStatic 消息处理程序\n\n\n\n\nvoid CColorStatic::OnPaint()\n{\n\tCPaintDC dc(this); // device context for painting\n\t\t\t\t\t   // TODO: 在此处添加消息处理程序代码\n\t\t\t\t\t   // 不为绘图消息调用 CStatic::OnPaint()\n\tCRect rect;\n\tGetClientRect(rect);\n\trect.MoveToXY(0, 0);\n\tCRect rc_tmp{ rect };\n\n\tCDrawCommon draw;\n\tdraw.Create(&dc, this);\n\n    int color_num = static_cast<int>(m_colors.size());\n\n\tif (IsWindowEnabled() && color_num > 0)\n\t{\n\t\tswitch (color_num)\n\t\t{\n\t\tcase 1:\n\t\t\tdc.FillSolidRect(rect, m_colors[0]);\n\t\t\tbreak;\n\t\tcase 4:\n\t\t\tdc.FillSolidRect(rect, RGB(255,255,255));\n\t\t\trc_tmp.right /= 2;\n\t\t\trc_tmp.bottom /= 2;\n\t\t\tdc.FillSolidRect(rc_tmp, m_colors[0]);\n\t\t\trc_tmp.MoveToX(rc_tmp.right);\n\t\t\tdc.FillSolidRect(rc_tmp, m_colors[1]);\n\t\t\trc_tmp.MoveToXY(0, rc_tmp.bottom);\n\t\t\tdc.FillSolidRect(rc_tmp, m_colors[2]);\n\t\t\trc_tmp.MoveToX(rc_tmp.right);\n\t\t\tdc.FillSolidRect(rc_tmp, m_colors[3]);\n\t\t\tbreak;\n\t\tcase 8:\n\t\t\tdc.FillSolidRect(rect, RGB(255, 255, 255));\n\t\t\trc_tmp.right /= (color_num / 2);\n\t\t\trc_tmp.bottom /= 2;\n\t\t\tdc.FillSolidRect(rc_tmp, m_colors[0]);\n\t\t\trc_tmp.MoveToX(rc_tmp.right);\n\t\t\tdc.FillSolidRect(rc_tmp, m_colors[1]);\n\t\t\trc_tmp.MoveToX(rc_tmp.right);\n\t\t\tdc.FillSolidRect(rc_tmp, m_colors[4]);\n\t\t\trc_tmp.MoveToX(rc_tmp.right);\n\t\t\tdc.FillSolidRect(rc_tmp, m_colors[5]);\n\t\t\trc_tmp.MoveToXY(0, rc_tmp.bottom);\n\t\t\tdc.FillSolidRect(rc_tmp, m_colors[2]);\n\t\t\trc_tmp.MoveToX(rc_tmp.right);\n\t\t\tdc.FillSolidRect(rc_tmp, m_colors[3]);\n\t\t\trc_tmp.MoveToX(rc_tmp.right);\n\t\t\tdc.FillSolidRect(rc_tmp, m_colors[6]);\n\t\t\trc_tmp.MoveToX(rc_tmp.right);\n\t\t\tdc.FillSolidRect(rc_tmp, m_colors[7]);\n\t\t\tbreak;\n        case 16:\n            dc.FillSolidRect(rect, RGB(255, 255, 255));\n            rc_tmp.right /= color_num / 2;\n            rc_tmp.bottom /= 2;\n            dc.FillSolidRect(rc_tmp, m_colors[0]);\n            rc_tmp.MoveToX(rc_tmp.right);\n            dc.FillSolidRect(rc_tmp, m_colors[1]);\n            rc_tmp.MoveToX(rc_tmp.right);\n            dc.FillSolidRect(rc_tmp, m_colors[4]);\n            rc_tmp.MoveToX(rc_tmp.right);\n            dc.FillSolidRect(rc_tmp, m_colors[5]);\n            rc_tmp.MoveToX(rc_tmp.right);\n            dc.FillSolidRect(rc_tmp, m_colors[8]);\n            rc_tmp.MoveToX(rc_tmp.right);\n            dc.FillSolidRect(rc_tmp, m_colors[9]);\n            rc_tmp.MoveToX(rc_tmp.right);\n            dc.FillSolidRect(rc_tmp, m_colors[12]);\n            rc_tmp.MoveToX(rc_tmp.right);\n            dc.FillSolidRect(rc_tmp, m_colors[13]);\n\n            rc_tmp.MoveToXY(0, rc_tmp.bottom);\n            dc.FillSolidRect(rc_tmp, m_colors[2]);\n            rc_tmp.MoveToX(rc_tmp.right);\n            dc.FillSolidRect(rc_tmp, m_colors[3]);\n            rc_tmp.MoveToX(rc_tmp.right);\n            dc.FillSolidRect(rc_tmp, m_colors[6]);\n            rc_tmp.MoveToX(rc_tmp.right);\n            dc.FillSolidRect(rc_tmp, m_colors[7]);\n            rc_tmp.MoveToX(rc_tmp.right);\n            dc.FillSolidRect(rc_tmp, m_colors[10]);\n            rc_tmp.MoveToX(rc_tmp.right);\n            dc.FillSolidRect(rc_tmp, m_colors[11]);\n            rc_tmp.MoveToX(rc_tmp.right);\n            dc.FillSolidRect(rc_tmp, m_colors[14]);\n            rc_tmp.MoveToX(rc_tmp.right);\n            dc.FillSolidRect(rc_tmp, m_colors[15]);\n            break;\n        default:\n\t\t\tdc.FillSolidRect(rect, RGB(255, 255, 255));\n\t\t\trc_tmp.right = rect.Width() / color_num;\n\t\t\tfor (int i{}; i < color_num; i++)\n\t\t\t{\n\t\t\t\trc_tmp.MoveToX(i*(rect.Width() / color_num));\n\t\t\t\tdc.FillSolidRect(rc_tmp, m_colors[i]);\n\t\t\t}\n\t\t}\n\n\t\t//画边框\n\t\tdraw.DrawRectOutLine(rect, RGB(160, 160, 160));\n\t}\n\telse\n\t{\n        CBrush brush(HS_BDIAGONAL, RGB(160, 160, 160));\n        dc.FillRect(rect, &brush);\n\t\tdraw.DrawRectOutLine(rect, RGB(192, 192, 192));\n\t}\n}\n\n\nvoid CColorStatic::OnMouseHover(UINT nFlags, CPoint point)\n{\n\t// TODO: 在此添加消息处理程序代码和/或调用默认值\n\tm_hover = true;\n\n\tCStatic::OnMouseHover(nFlags, point);\n}\n\n\nvoid CColorStatic::OnMouseLeave()\n{\n\t// TODO: 在此添加消息处理程序代码和/或调用默认值\n\tm_hover = false;\n\n\tCStatic::OnMouseLeave();\n}\n\n\nBOOL CColorStatic::OnSetCursor(CWnd* pWnd, UINT nHitTest, UINT message)\n{\n\t// TODO: 在此添加消息处理程序代码和/或调用默认值\n\tif(m_link_cursor && m_hover)\n\t{\n\t\t::SetCursor(::LoadCursor(NULL, MAKEINTRESOURCE(32649)));\n\t\treturn TRUE;\n\t}\n\n\treturn CStatic::OnSetCursor(pWnd, nHitTest, message);\n}\n\n\nvoid CColorStatic::PreSubclassWindow()\n{\n\t// TODO: 在此添加专用代码和/或调用基类\n\tDWORD dwStyle = GetStyle();\n\t::SetWindowLong(GetSafeHwnd(), GWL_STYLE, dwStyle | SS_NOTIFY);\n\tModifyStyleEx(WS_EX_STATICEDGE, NULL);\n\n\tCStatic::PreSubclassWindow();\n}\n\n\nvoid CColorStatic::OnMouseMove(UINT nFlags, CPoint point)\n{\n\t// TODO: 在此添加消息处理程序代码和/或调用默认值\n\tif (m_link_cursor)\n\t{\n\t\tTRACKMOUSEEVENT tme;\n\t\ttme.cbSize = sizeof(tme);\n\t\ttme.hwndTrack = m_hWnd;\n\t\ttme.dwFlags = TME_LEAVE | TME_HOVER;\n\t\ttme.dwHoverTime = 1;\n\t\t_TrackMouseEvent(&tme);\n\t}\n\n\tCStatic::OnMouseMove(nFlags, point);\n}\n\n\nvoid CColorStatic::OnLButtonUp(UINT nFlags, CPoint point)\n{\n\t// TODO: 在此添加消息处理程序代码和/或调用默认值\n\t//如果单击了鼠标左键，则向父窗口发送一个WM_STATIC_CLICKED消息\n\tCWnd* pParent{ GetParent() };\n\tif (pParent != nullptr)\n\t\tpParent->SendMessage(WM_STATIC_CLICKED, (WPARAM)this);\n\n\tCStatic::OnLButtonUp(nFlags, point);\n}\n"
  },
  {
    "path": "TrafficMonitor/ColorStatic.h",
    "content": "﻿// CColorStatic\n//派生自CStatic，用于显示颜色的Static控件\n#pragma once\n#define WM_STATIC_CLICKED (WM_USER + 1001)\n\nclass CColorStatic : public CStatic\n{\n\tDECLARE_DYNAMIC(CColorStatic)\n\npublic:\n\tCColorStatic();\n\tvirtual ~CColorStatic();\n\n\tvoid SetFillColor(COLORREF fill_color);\t\t//设置要填充单一的背景色\n\tvoid SetColorNum(int color_num);\t\t\t//设置颜色的数量\n\tvoid SetFillColor(int index, COLORREF fill_color);\n\tvoid SetLinkCursor(bool link_cursor = true);\t\t//设置指向控件时光标变成超链接形状\n    void EnableWindow(bool enable);\n\nprotected:\n\t//COLORREF m_fill_color{ RGB(255, 255,255) };\n\tvector<COLORREF> m_colors;\n\tbool m_hover{ false };\t\t//当鼠标指向控件时为true\n\tbool m_link_cursor{ false };\t//是否启用超链接形状的光标\n\nprotected:\n\tDECLARE_MESSAGE_MAP()\npublic:\n\tafx_msg void OnPaint();\n\tafx_msg void OnMouseHover(UINT nFlags, CPoint point);\n\tafx_msg void OnMouseLeave();\n\tafx_msg BOOL OnSetCursor(CWnd* pWnd, UINT nHitTest, UINT message);\n\tvirtual void PreSubclassWindow();\n\tafx_msg void OnMouseMove(UINT nFlags, CPoint point);\n\tafx_msg void OnLButtonUp(UINT nFlags, CPoint point);\n};\n"
  },
  {
    "path": "TrafficMonitor/ComboBox2.cpp",
    "content": "﻿// ComboBox2.cpp: 实现文件\n//\n\n#include \"stdafx.h\"\n#include \"ComboBox2.h\"\n\n\n// CComboBox2\n\nIMPLEMENT_DYNAMIC(CComboBox2, CComboBox)\n\nCComboBox2::CComboBox2()\n{\n\n}\n\nCComboBox2::~CComboBox2()\n{\n}\n\n\nvoid CComboBox2::SetMouseWheelEnable(bool enable)\n{\n    m_mouse_wheel_enable = enable;\n}\n\nBEGIN_MESSAGE_MAP(CComboBox2, CComboBox)\nEND_MESSAGE_MAP()\n\n\n\n// CComboBox2 消息处理程序\n\n\n\n\nBOOL CComboBox2::PreTranslateMessage(MSG* pMsg)\n{\n    // TODO: 在此添加专用代码和/或调用基类\n    //如果m_mouse_wheel_enable为false，则不响应鼠标滚轮消息\n    if (pMsg->message == WM_MOUSEWHEEL && !m_mouse_wheel_enable)\n    {\n        //将鼠标滚轮消息发送给父窗口\n        CWnd* pParent = GetParent();\n        pParent->SendMessage(WM_MOUSEWHEEL, pMsg->wParam, pMsg->lParam);\n        return true;\n    }\n\n    return CComboBox::PreTranslateMessage(pMsg);\n}\n"
  },
  {
    "path": "TrafficMonitor/ComboBox2.h",
    "content": "﻿#pragma once\n\n\n// CComboBox2\n\nclass CComboBox2 : public CComboBox\n{\n\tDECLARE_DYNAMIC(CComboBox2)\n\npublic:\n\tCComboBox2();\n\tvirtual ~CComboBox2();\n    void SetMouseWheelEnable(bool enable);  //设置是否允许响应鼠标滚轮\n\nprivate:\n    bool m_mouse_wheel_enable{ true };\n\nprotected:\n\tDECLARE_MESSAGE_MAP()\npublic:\n    virtual BOOL PreTranslateMessage(MSG* pMsg);\n};\n\n\n"
  },
  {
    "path": "TrafficMonitor/Common.cpp",
    "content": "﻿#include \"stdafx.h\"\n#include \"Common.h\"\n#include \"TrafficMonitor.h\"\n\n\nCCommon::CCommon()\n{\n}\n\n\nCCommon::~CCommon()\n{\n}\n\nwstring CCommon::StrToUnicode(const char* str, bool utf8)\n{\n    wstring result;\n    int size;\n    size = MultiByteToWideChar((utf8 ? CP_UTF8 : CP_ACP), 0, str, -1, NULL, 0);\n    if (size <= 0) return wstring();\n    wchar_t* str_unicode = new wchar_t[size + 1];\n    MultiByteToWideChar((utf8 ? CP_UTF8 : CP_ACP), 0, str, -1, str_unicode, size);\n    result.assign(str_unicode);\n    delete[] str_unicode;\n    return result;\n}\n\nstring CCommon::UnicodeToStr(const wchar_t* wstr, bool utf8)\n{\n    string result;\n    int size{ 0 };\n    size = WideCharToMultiByte((utf8 ? CP_UTF8 : CP_ACP), 0, wstr, -1, NULL, 0, NULL, NULL);\n    if (size <= 0) return string();\n    char* str = new char[size + 1];\n    WideCharToMultiByte((utf8 ? CP_UTF8 : CP_ACP), 0, wstr, -1, str, size, NULL, NULL);\n    result.assign(str);\n    delete[] str;\n    return result;\n}\n\nbool CCommon::GetFileContent(const wchar_t* file_path, string& contents_buff, bool binary /*= true*/)\n{\n    std::ifstream file{ file_path, (binary ? std::ios::binary : std::ios::in) };\n    if (file.fail())\n        return false;\n    //获取文件长度\n    file.seekg(0, file.end);\n    size_t length = file.tellg();\n    file.seekg(0, file.beg);\n\n    char* buff = new char[length];\n    file.read(buff, length);\n    file.close();\n\n    contents_buff.assign(buff, length);\n    delete[] buff;\n\n    return true;\n}\n\nconst char* CCommon::GetFileContent(const wchar_t* file_path, size_t& length, bool binary /*= true*/)\n{\n    std::ifstream file{ file_path, (binary ? std::ios::binary : std::ios::in) };\n    length = 0;\n    if (file.fail())\n        return nullptr;\n    //获取文件长度\n    file.seekg(0, file.end);\n    length = file.tellg();\n    file.seekg(0, file.beg);\n\n    char* buff = new char[length];\n    file.read(buff, length);\n    file.close();\n\n    return buff;\n}\n\nCString CCommon::DataSizeToString(unsigned long long size, const PublicSettingData& cfg)\n{\n    //CString str;\n    CString value_str, unit_str;\n    if (!cfg.unit_byte)     //如果使用比特(bit)为单位，则数值乘以8\n    {\n        size *= 8;\n    }\n    switch (cfg.speed_unit)\n    {\n    case SpeedUnit::AUTO:\n        if (cfg.speed_short_mode)\n        {\n            if (size < 1024 * 10)                   //10KB以下以KB为单位，保留1位小数\n            {\n                value_str.Format(_T(\"%.1f\"), size / 1024.0f);\n                unit_str = _T(\"K\");\n            }\n            else if (size < 1024 * 1000)            //1000KB以下以KB为单位，保留整数\n            {\n                value_str.Format(_T(\"%.0f\"), size / 1024.0f);\n                unit_str = _T(\"K\");\n            }\n            else if (size < 1024 * 1024 * 1000)     //1000MB以下以MB为单位，保留1位小数\n            {\n                value_str.Format(_T(\"%.1f\"), size / 1024.0f / 1024.0f);\n                unit_str = _T(\"M\");\n            }\n            else\n            {\n                value_str.Format(_T(\"%.2f\"), size / 1024.0f / 1024.0f / 1024.0f);\n                unit_str = _T(\"G\");\n            }\n        }\n        else\n        {\n            if (size < 1024 * 10)                   //10KB以下以KB为单位，保留2位小数\n            {\n                value_str.Format(_T(\"%.2f\"), size / 1024.0f);\n                unit_str = _T(\"KB\");\n            }\n            else if (size < 1024 * 1000)            //1000KB以下以KB为单位，保留1位小数\n            {\n                value_str.Format(_T(\"%.1f\"), size / 1024.0f);\n                unit_str = _T(\"KB\");\n            }\n            else if (size < 1024 * 1024 * 1000)     //1000MB以下以MB为单位，保留2位小数\n            {\n                value_str.Format(_T(\"%.2f\"), size / 1024.0f / 1024.0f);\n                unit_str = _T(\"MB\");\n            }\n            else\n            {\n                value_str.Format(_T(\"%.2f\"), size / 1024.0f / 1024.0f / 1024.0f);\n                unit_str = _T(\"GB\");\n            }\n        }\n        break;\n    case SpeedUnit::KBPS:\n        if (cfg.speed_short_mode)\n        {\n            if (size < 1024 * 10)                   //10KB以下保留1位小数\n                value_str.Format(_T(\"%.1f\"), size / 1024.0f);\n            else                    //10KB以上保留整数\n                value_str.Format(_T(\"%.0f\"), size / 1024.0f);\n            if (!cfg.hide_unit)\n                unit_str = _T(\"K\");\n        }\n        else\n        {\n            if (size < 1024 * 10)                   //10KB以下保留2位小数\n                value_str.Format(_T(\"%.2f\"), size / 1024.0f);\n            else            //10KB以上保留1位小数\n                value_str.Format(_T(\"%.1f\"), size / 1024.0f);\n            if (!cfg.hide_unit)\n                unit_str = _T(\"KB\");\n        }\n        break;\n    case SpeedUnit::MBPS:\n        if (cfg.speed_short_mode)\n        {\n            value_str.Format(_T(\"%.1f\"), size / 1024.0f / 1024.0f);\n            if (!cfg.hide_unit)\n                unit_str = _T(\"M\");\n        }\n        else\n        {\n            value_str.Format(_T(\"%.2f\"), size / 1024.0f / 1024.0f);\n            if (!cfg.hide_unit)\n                unit_str = _T(\"MB\");\n        }\n        break;\n    }\n    CString str;\n    if (cfg.separate_value_unit_with_space && !cfg.hide_unit)\n        str = value_str + _T(' ') + unit_str;\n    else\n        str = value_str + unit_str;\n    if (!cfg.unit_byte)\n    {\n        if (cfg.speed_short_mode && !cfg.hide_unit)\n            str += _T('b');     //如果使用比特(bit)为单位，即使设置了网速简洁模式，也将“b”显示出来\n        else\n            str.Replace(_T('B'), _T('b'));  //如果使用比特(bit)为单位，将B替换成b\n    }\n    return str;\n}\n\nCString CCommon::DataSizeToString(unsigned long long size, bool with_space)\n{\n    CString str;\n    if (size < 1024 * 10)                   //10KB以下以KB为单位，保留2位小数\n        str.Format(_T(\"%.2f KB\"), size / 1024.0);\n    else if (size < 1024 * 1024)            //1MB以下以KB为单位，保留1位小数\n        str.Format(_T(\"%.1f KB\"), size / 1024.0);\n    else if (size < 1024 * 1024 * 1024)     //1GB以下以MB为单位，保留2位小数\n        str.Format(_T(\"%.2f MB\"), size / 1024.0 / 1024.0);\n    else if (size < 1024ll * 1024 * 1024 * 1024)\n        str.Format(_T(\"%.2f GB\"), size / 1024.0 / 1024.0 / 1024.0);\n    else\n        str.Format(_T(\"%.2f TB\"), size / 1024.0 / 1024.0 / 1024.0 / 1024.0);\n    if (!with_space)\n        str.Remove(_T(' '));\n    return str;\n}\n\nCString CCommon::TemperatureToString(float temperature, const PublicSettingData& cfg)\n{\n    CString str_val;\n    if (temperature <= 0)\n        str_val = _T(\"--\");\n    else\n        str_val.Format(_T(\"%d\"), static_cast<int>(temperature));\n    if (cfg.separate_value_unit_with_space)\n        str_val += _T(' ');\n    str_val += _T(\"°C\");\n    return str_val;\n}\n\nCString CCommon::UsageToString(int usage, const PublicSettingData& cfg)\n{\n    CString str_val;\n    if (usage < 0)\n        str_val = _T(\"--\");\n    else\n        str_val.Format(_T(\"%d\"), usage);\n    if (!cfg.hide_percent)\n    {\n        if (cfg.separate_value_unit_with_space)\n            str_val += _T(' ');\n        str_val += _T('%');\n    }\n    return str_val;\n}\n\nCString CCommon::FreqToString(float freq, const PublicSettingData& cfg)\n{\n    CString str_val;\n    if (freq < 0)\n        str_val = _T(\"--\");\n    else\n        str_val.Format(_T(\"%.2f GHz\"), freq);\n    return str_val;\n}\n//CString CCommon::KBytesToString(unsigned int kb_size)\n//{\n//  CString k_bytes_str;\n//  if (kb_size < 1024)\n//      k_bytes_str.Format(_T(\"%d KB\"), kb_size);\n//  else if (kb_size < 1024 * 1024)\n//      k_bytes_str.Format(_T(\"%.2f MB\"), kb_size / 1024.0);\n//  else if (kb_size < 1024 * 1024 * 1024)\n//      k_bytes_str.Format(_T(\"%.2f GB\"), kb_size / 1024.0 / 1024.0);\n//  else\n//      k_bytes_str.Format(_T(\"%.2f TB\"), kb_size / 1024.0 / 1024.0 / 1024.0);\n//  return k_bytes_str;\n//}\n\nCString CCommon::KBytesToString(unsigned __int64 kb_size)\n{\n    CString k_bytes_str;\n    if (kb_size < 1024)\n        k_bytes_str.Format(_T(\"%d KB\"), kb_size);\n    else if (kb_size < 1024 * 1024)\n        k_bytes_str.Format(_T(\"%.2f MB\"), kb_size / 1024.0);\n    else if (kb_size < 1024 * 1024 * 1024)\n        k_bytes_str.Format(_T(\"%.2f GB\"), kb_size / 1024.0 / 1024.0);\n    else\n        k_bytes_str.Format(_T(\"%.2f TB\"), kb_size / 1024.0 / 1024.0 / 1024.0);\n    return k_bytes_str;\n}\n\n__int64 CCommon::CompareFileTime2(FILETIME time1, FILETIME time2)\n{\n    __int64 a = static_cast<__int64>(time1.dwHighDateTime) << 32 | time1.dwLowDateTime;\n    __int64 b = static_cast<__int64>(time2.dwHighDateTime) << 32 | time2.dwLowDateTime;\n    return b - a;\n}\n\nvoid CCommon::WriteLog(const char* str_text, LPCTSTR file_path)\n{\n    SYSTEMTIME cur_time;\n    GetLocalTime(&cur_time);\n    char buff[32];\n    sprintf_s(buff, \"%d/%.2d/%.2d %.2d:%.2d:%.2d.%.3d: \", cur_time.wYear, cur_time.wMonth, cur_time.wDay,\n        cur_time.wHour, cur_time.wMinute, cur_time.wSecond, cur_time.wMilliseconds);\n    ofstream file{ file_path, std::ios::app };  //以追加的方式打开日志文件\n    file << buff;\n    file << str_text << std::endl;\n}\n\nvoid CCommon::WriteLog(const wchar_t* str_text, LPCTSTR file_path)\n{\n    SYSTEMTIME cur_time;\n    GetLocalTime(&cur_time);\n    char buff[32];\n    sprintf_s(buff, \"%d/%.2d/%.2d %.2d:%.2d:%.2d.%.3d: \", cur_time.wYear, cur_time.wMonth, cur_time.wDay,\n        cur_time.wHour, cur_time.wMinute, cur_time.wSecond, cur_time.wMilliseconds);\n    ofstream file{ file_path, std::ios::app };  //以追加的方式打开日志文件\n    file << buff;\n    file << UnicodeToStr(str_text).c_str() << std::endl;\n}\n\nBOOL CCommon::CreateFileShortcut(LPCTSTR lpszLnkFileDir, LPCTSTR lpszFileName, LPCTSTR lpszLnkFileName, LPCTSTR lpszWorkDir, WORD wHotkey, LPCTSTR lpszDescription, int iShowCmd)\n{\n    if (lpszLnkFileDir == NULL)\n        return FALSE;\n\n    HRESULT hr;\n    IShellLink* pLink;  //IShellLink对象指针\n    IPersistFile* ppf; //IPersisFil对象指针\n\n                         //创建IShellLink对象\n    hr = CoCreateInstance(CLSID_ShellLink, NULL, CLSCTX_INPROC_SERVER, IID_IShellLink, (void**)&pLink);\n    if (FAILED(hr))\n        return FALSE;\n\n    //从IShellLink对象中获取IPersistFile接口\n    hr = pLink->QueryInterface(IID_IPersistFile, (void**)&ppf);\n    if (FAILED(hr))\n    {\n        pLink->Release();\n        return FALSE;\n    }\n\n    TCHAR file_path[MAX_PATH];\n    GetModuleFileName(NULL, file_path, MAX_PATH);\n\n    //目标\n    if (lpszFileName == NULL)\n        pLink->SetPath(file_path);\n    else\n        pLink->SetPath(lpszFileName);\n\n    //工作目录\n    if (lpszWorkDir != NULL)\n    {\n        pLink->SetWorkingDirectory(lpszWorkDir);\n    }\n    else\n    {\n        //设置工作目录为快捷方式目标所在位置\n        TCHAR workDirBuf[MAX_PATH]{};\n        if (lpszFileName == NULL)\n            //wcscpy_s(workDirBuf, file_path);\n            WStringCopy(workDirBuf, 260, file_path, 260);\n        else\n            //wcscpy_s(workDirBuf, lpszFileName);\n            WStringCopy(workDirBuf, 260, lpszFileName);\n        LPTSTR pstr = wcsrchr(workDirBuf, _T('\\\\'));\n        *pstr = _T('\\0');\n        pLink->SetWorkingDirectory(workDirBuf);\n    }\n\n    //快捷键\n    if (wHotkey != 0)\n        pLink->SetHotkey(wHotkey);\n\n    //备注\n    if (lpszDescription != NULL)\n        pLink->SetDescription(lpszDescription);\n\n    //显示方式\n    pLink->SetShowCmd(iShowCmd);\n\n\n    //快捷方式的路径 + 名称\n    wchar_t szBuffer[MAX_PATH];\n    if (lpszLnkFileName != NULL) //指定了快捷方式的名称\n        swprintf_s(szBuffer, L\"%s\\\\%s\", lpszLnkFileDir, lpszLnkFileName);\n    else\n    {\n        //没有指定名称，就从取指定文件的文件名作为快捷方式名称。\n        const wchar_t* pstr;\n        if (lpszFileName != NULL)\n            pstr = wcsrchr(lpszFileName, L'\\\\');\n        else\n            pstr = wcsrchr(file_path, L'\\\\');\n\n        if (pstr == NULL)\n        {\n            ppf->Release();\n            pLink->Release();\n            return FALSE;\n        }\n        //注意后缀名要从.exe改为.lnk\n        swprintf_s(szBuffer, L\"%s\\\\%s\", lpszLnkFileDir, pstr);\n        int nLen = wcslen(szBuffer);\n        szBuffer[nLen - 3] = L'l';\n        szBuffer[nLen - 2] = L'n';\n        szBuffer[nLen - 1] = L'k';\n    }\n    //保存快捷方式到指定目录下\n    //WCHAR  wsz[MAX_PATH];  //定义Unicode字符串\n    //MultiByteToWideChar(CP_ACP, 0, szBuffer, -1, wsz, MAX_PATH);\n\n    hr = ppf->Save(szBuffer, TRUE);\n\n    ppf->Release();\n    pLink->Release();\n    return SUCCEEDED(hr);\n}\n\nwstring CCommon::GetStartUpPath()\n{\n    LPITEMIDLIST ppidl;\n    TCHAR pszStartUpPath[MAX_PATH]{};\n    if (SHGetSpecialFolderLocation(NULL, CSIDL_STARTUP, &ppidl) == S_OK)\n    {\n        SHGetPathFromIDList(ppidl, pszStartUpPath);\n        CoTaskMemFree(ppidl);\n    }\n    return wstring(pszStartUpPath);\n}\n\nvoid CCommon::GetFiles(const wchar_t* path, vector<wstring>& files)\n{\n    //文件句柄\n    intptr_t hFile = 0;\n    //文件信息（用Unicode保存使用_wfinddata_t，多字节字符集使用_finddata_t）\n    _wfinddata_t fileinfo;\n    wstring file_name;\n    if ((hFile = _wfindfirst(path, &fileinfo)) != -1)\n    {\n        do\n        {\n            file_name.assign(fileinfo.name);\n            if (file_name != L\".\" && file_name != L\"..\")\n                //files.push_back(wstring(path) + L\"\\\\\" + file_name);  //将文件名保存(忽略\".\"和\"..\")\n                files.push_back(L\"\\\\\" + file_name);  //将文件名保存(忽略\".\"和\"..\")\n        } while (_wfindnext(hFile, &fileinfo) == 0);\n    }\n    _findclose(hFile);\n}\n\nvoid CCommon::GetFiles(const wchar_t* path, std::function<void(const wstring&)> func)\n{\n    //文件句柄\n    intptr_t hFile = 0;\n    _wfinddata_t fileinfo;\n    wstring file_name;\n    if ((hFile = _wfindfirst(path, &fileinfo)) != -1)\n    {\n        do\n        {\n            file_name.assign(fileinfo.name);\n            if (file_name != L\".\" && file_name != L\"..\")\n                func(file_name);\n        } while (_wfindnext(hFile, &fileinfo) == 0);\n    }\n    _findclose(hFile);\n}\n\nbool CCommon::FileExist(LPCTSTR file_name)\n{\n    return (PathFileExists(file_name) != 0);\n}\n\nbool CCommon::IsFolder(const wstring& path)\n{\n    DWORD dwAttrib = GetFileAttributes(path.c_str());\n    return (dwAttrib & FILE_ATTRIBUTE_DIRECTORY) != 0;\n}\n\nbool CCommon::MoveAFile(LPCTSTR exist_file, LPCTSTR new_file)\n{\n    if (!FileExist(exist_file))\n        return false;\n    //if (FileExist(new_file))      //如果目标文件已经存在，则先删除它\n    //  DeleteFile(new_file);\n    return (MoveFile(exist_file, new_file) != 0);\n}\n\nSYSTEMTIME CCommon::CompareSystemTime(SYSTEMTIME a, SYSTEMTIME b)\n{\n    SYSTEMTIME result{};\n    short hour = a.wHour - b.wHour;\n    short minute = a.wMinute - b.wMinute;\n    short second = a.wSecond - b.wSecond;\n\n    if (second < 0)\n    {\n        second += 60;\n        minute--;\n    }\n\n    if (minute < 0)\n    {\n        minute += 60;\n        hour--;\n    }\n\n    if (hour < 0)\n    {\n        hour += 24;\n    }\n    result.wHour = hour;\n    result.wMinute = minute;\n    result.wSecond = second;\n    return result;\n}\n\nwstring CCommon::GetModuleDir()\n{\n    wchar_t path[MAX_PATH];\n    GetModuleFileNameW(NULL, path, MAX_PATH);\n    size_t index;\n    wstring current_path{ path };\n    index = current_path.find_last_of(L'\\\\');\n    current_path = current_path.substr(0, index + 1);\n    return current_path;\n}\n\nwstring CCommon::GetSystemDir()\n{\n    wchar_t buff[MAX_PATH];\n    GetSystemDirectory(buff, MAX_PATH);\n    return wstring(buff);\n}\n\nwstring CCommon::GetTemplateDir()\n{\n    wstring result;\n    wchar_t buff[MAX_PATH];\n    GetTempPath(MAX_PATH, buff);        //获取临时文件夹的路径\n    result = buff;\n    if (result.back() != L'\\\\' && result.back() != L'/')        //确保路径后面有斜杠\n        result.push_back(L'\\\\');\n    return result;\n}\n\nwstring CCommon::GetAppDataConfigDir()\n{\n    LPITEMIDLIST ppidl;\n    TCHAR pszAppDataPath[MAX_PATH];\n    if (SHGetSpecialFolderLocation(NULL, CSIDL_APPDATA, &ppidl) == S_OK)\n    {\n        SHGetPathFromIDList(ppidl, pszAppDataPath);\n        CoTaskMemFree(ppidl);\n    }\n    wstring app_data_path{ pszAppDataPath };        //获取到C:/User/用户名/AppData/Roaming路径\n    CreateDirectory(app_data_path.c_str(), NULL);       //如果Roaming不存在，则创建它\n    app_data_path += L\"\\\\TrafficMonitor\\\\\";\n    CreateDirectory(app_data_path.c_str(), NULL);       //如果C:/User/用户名/AppData/Roaming/TrafficMonitor不存在，则创建它\n\n    return app_data_path;\n}\n\nvoid CCommon::DrawWindowText(CDC* pDC, CRect rect, LPCTSTR lpszString, COLORREF color, COLORREF back_color)\n{\n    pDC->SetTextColor(color);\n    //m_pDC->SetBkMode(TRANSPARENT);\n    //用背景色填充矩形区域\n    pDC->FillSolidRect(rect, back_color);\n    pDC->DrawText(lpszString, rect, DT_VCENTER | DT_SINGLELINE | DT_NOPREFIX);\n\n}\n\n//void CCommon::SetDrawArea(CDC * pDC, CRect rect)\n//{\n//  CRgn rgn;\n//  rgn.CreateRectRgnIndirect(rect);\n//  pDC->SelectClipRgn(&rgn);\n//}\n\n\nbool CCommon::IsForegroundFullscreen()\n{\n    bool bFullscreen{ false };      //用于指示前台窗口是否是全屏\n    HWND hWnd;\n    RECT rcApp;\n    RECT rcDesk;\n    hWnd = GetForegroundWindow();   //获取当前正在与用户交互的前台窗口句柄\n    TCHAR buff[256];\n    GetClassName(hWnd, buff, 256);      //获取前台窗口的类名\n    CString class_name{ buff };\n    if (hWnd != GetDesktopWindow() && class_name != _T(\"WorkerW\") && hWnd != GetShellWindow())//如果前台窗口不是桌面窗口，也不是控制台窗口\n    {\n        GetWindowRect(hWnd, &rcApp);    //获取前台窗口的坐标\n        GetWindowRect(GetDesktopWindow(), &rcDesk); //根据桌面窗口句柄，获取整个屏幕的坐标\n        if (rcApp.left <= rcDesk.left && //如果前台窗口的坐标完全覆盖住桌面窗口，就表示前台窗口是全屏的\n            rcApp.top <= rcDesk.top &&\n            rcApp.right >= rcDesk.right &&\n            rcApp.bottom >= rcDesk.bottom)\n        {\n            bFullscreen = true;\n        }\n    }//如果前台窗口是桌面窗口，或者是控制台窗口，就直接返回不是全屏\n    return bFullscreen;\n}\n\nbool CCommon::CopyStringToClipboard(const wstring& str)\n{\n    if (OpenClipboard(NULL))\n    {\n        HGLOBAL clipbuffer;\n        EmptyClipboard();\n        size_t size = (str.size() + 1) * 2;\n        clipbuffer = GlobalAlloc(GMEM_DDESHARE, size);\n        memcpy_s(GlobalLock(clipbuffer), size, str.c_str(), size);\n        GlobalUnlock(clipbuffer);\n        if (SetClipboardData(CF_UNICODETEXT, clipbuffer) == NULL)\n            return false;\n        CloseClipboard();\n        return true;\n    }\n    else return false;\n}\n\n\nwstring CCommon::GetJsonValueSimple(const wstring& json_str, const wstring& name)\n{\n    wstring str_name{ L\"\\\"\" };\n    str_name += name;\n    str_name += L'\\\"';\n    size_t index = json_str.find(str_name);\n    if (index == wstring::npos)\n        return wstring();\n    index = json_str.find(L':', index + 1);\n    if (index == wstring::npos)\n        return wstring();\n    index = json_str.find_first_not_of(L\"\\\" \", index + 1);\n    size_t index_end = json_str.find_first_of(L\"\\\",]}\\r\\n\", index);\n    wstring result = json_str.substr(index, index_end - index);\n    return result;\n}\n\nbool CCommon::GetURL(const wstring& url, wstring& result, bool utf8, const wstring& user_agent)\n{\n    bool succeed{ false };\n    CInternetSession* pSession{};\n    CHttpFile* pfile{};\n    try\n    {\n        pSession = new CInternetSession(user_agent.c_str());\n        pfile = (CHttpFile*)pSession->OpenURL(url.c_str());\n        DWORD dwStatusCode;\n        pfile->QueryInfoStatusCode(dwStatusCode);\n        if (dwStatusCode == HTTP_STATUS_OK)\n        {\n            CString content;\n            CString data;\n            while (pfile->ReadString(data))\n            {\n                content += data;\n            }\n            result = StrToUnicode((const char*)content.GetString(), utf8);\n            succeed = true;\n        }\n        pfile->Close();\n        delete pfile;\n        pSession->Close();\n    }\n    catch (CInternetException* e)\n    {\n        //写入错误日志\n        if (theApp.m_debug_log)\n        {\n            CString info = CCommon::LoadTextFormat(IDS_GET_URL_ERROR_LOG_INFO, { url, static_cast<size_t>(e->m_dwError) });\n            CCommon::WriteLog(info, theApp.m_log_path.c_str());\n        }\n        if (pfile != nullptr)\n        {\n            pfile->Close();\n            delete pfile;\n        }\n        if (pSession != nullptr)\n            pSession->Close();\n        succeed = false;\n        e->Delete();        //没有这句会造成内存泄露\n        SAFE_DELETE(pSession);\n    }\n    SAFE_DELETE(pSession);\n    return succeed;\n}\n\n\nvoid CCommon::GetInternetIp(wstring& ip_address, wstring& ip_location, bool global)\n{\n    wstring web_page;\n    if (GetURL(L\"https://ip.cn/\", web_page, true))\n    {\n#ifdef _DEBUG\n        ofstream file{ L\".\\\\IP_web_page.log\" };\n        file << UnicodeToStr(web_page.c_str()) << std::endl;\n#endif // _DEBUG\n        size_t index, index1;\n        index = web_page.find(L\"<code>\");\n        index1 = web_page.find(L\"</code>\", index + 6);\n        if (index == wstring::npos || index1 == wstring::npos)\n            ip_address.clear();\n        else\n            ip_address = web_page.substr(index + 6, index1 - index - 6);    //获取IP地址\n        if (ip_address.size() > 15 || ip_address.size() < 7)        //IP地址最长15个字符，最短7个字符\n            ip_address.clear();\n\n        //获取IP地址归属地\n        if (!global)\n        {\n            index = web_page.find(L\"<code>\", index1 + 7);\n            index1 = web_page.find(L\"</code>\", index + 6);\n            if (index == wstring::npos || index1 == wstring::npos)\n                ip_location.clear();\n            else\n                ip_location = web_page.substr(index + 6, index1 - index - 6);\n        }\n        else\n        {\n            index = web_page.find(L\"GeoIP\", index1 + 7);\n            index1 = web_page.find(L\"</p>\", index + 6);\n            if (index == wstring::npos || index1 == wstring::npos)\n                ip_location.clear();\n            else\n                ip_location = web_page.substr(index + 7, index1 - index - 7);\n        }\n    }\n    else\n    {\n        ip_address.clear();\n    }\n}\n\nvoid CCommon::GetInternetIp2(wstring& ip_address, wstring& ip_location, bool ipv6)\n{\n    wstring raw_string;\n    wstring user_agent{ L\"TrafficMonitor/\" };\n    user_agent += VERSION;\n    if (GetURL((ipv6 ? L\"https://v6.yinghualuo.cn/bejson\" : L\"https://v4.yinghualuo.cn/bejson\"), raw_string, true, user_agent))\n    {\n        //解析获取的json字符串\n        ip_address = GetJsonValueSimple(raw_string, L\"ip\");\n        ip_location = GetJsonValueSimple(raw_string, L\"location\");\n\n    }\n    else\n    {\n        ip_address.clear();\n        ip_location.clear();\n    }\n}\n\n\nvoid CCommon::SetRect(CRect& rect, int x, int y, int width, int height)\n{\n    rect.left = x;\n    rect.top = y;\n    rect.right = x + width;\n    rect.bottom = y + height;\n}\n\nCString CCommon::LoadText(UINT id, LPCTSTR back_str)\n{\n    CString str;\n    str.LoadString(id);\n    if (back_str != nullptr)\n        str += back_str;\n    return str;\n}\n\nCString CCommon::LoadText(LPCTSTR front_str, UINT id, LPCTSTR back_str)\n{\n    CString str;\n    str.LoadString(id);\n    if (back_str != nullptr)\n        str += back_str;\n    if (front_str != nullptr)\n        str = front_str + str;\n    return str;\n}\n\nCString CCommon::StringFormat(LPCTSTR format_str, const std::initializer_list<CVariant>& paras)\n{\n    CString str_rtn = format_str;\n    int index = 1;\n    for (const auto& para : paras)\n    {\n        CString para_str = para.ToString();\n        CString format_para;\n        format_para.Format(_T(\"<%%%d%%>\"), index);\n        str_rtn.Replace(format_para, para_str);\n\n        index++;\n    }\n    return str_rtn;\n}\n\nCString CCommon::LoadTextFormat(UINT id, const std::initializer_list<CVariant>& paras)\n{\n    CString str;\n    str.LoadString(id);\n    return StringFormat(str.GetString(), paras);\n}\n\nCString CCommon::IntToString(__int64 n, bool thousand_separation, bool is_unsigned)\n{\n    wstring str = std::to_wstring(is_unsigned ? static_cast<unsigned __int64>(n) : n);\n    int count{};\n    if (thousand_separation)\n    {\n        int length{ static_cast<int>(str.size()) };\n        for (int i{ length - 1 }; i > 0; i--)\n        {\n            count++;\n            if (count % 3 == 0)\n                str.insert(i, L\",\");\n        }\n    }\n    return str.c_str();\n}\n\nvoid CCommon::NormalizeFont(LOGFONT& font)\n{\n    wstring name;\n    wstring style;\n    name = font.lfFaceName;\n    if (name.empty())\n        return;\n    if (name.back() == L' ')\n        name.pop_back();\n    size_t index = name.rfind(L' ');\n    if (index == wstring::npos)\n        return;\n    style = name.substr(index + 1);\n    bool style_acquired = false;\n    if (style == L\"Light\")\n    {\n        font.lfWeight = FW_LIGHT;\n        style_acquired = true;\n    }\n    else if (style == L\"Semilight\")\n    {\n        font.lfWeight = 350;\n        style_acquired = true;\n    }\n    else if (style == L\"Semibold\")\n    {\n        font.lfWeight = FW_SEMIBOLD;\n        style_acquired = true;\n    }\n    else if (style == L\"Bold\")\n    {\n        font.lfWeight = FW_BOLD;\n        style_acquired = true;\n    }\n    else if (style == L\"Black\")\n    {\n        font.lfWeight = FW_BLACK;\n        style_acquired = true;\n    }\n\n    if (style_acquired)\n    {\n        name = name.substr(0, index);\n    }\n    //wcsncpy_s(font.lfFaceName, name.c_str(), 32);\n    WStringCopy(font.lfFaceName, 32, name.c_str(), name.size());\n}\n\nvoid CCommon::WStringCopy(wchar_t* str_dest, int dest_size, const wchar_t* str_source, int source_size)\n{\n    if (dest_size <= 0)\n        return;\n    if (source_size <= 0 || str_source == nullptr)\n    {\n        str_dest[0] = L'\\0';\n        return;\n    }\n    int i;\n    for (i = 0; i < dest_size && i < source_size && str_source[i] != L'\\0'; i++)\n        str_dest[i] = str_source[i];\n    //确保目标字符串末尾有一个\\0\n    int copy_cnt = i;\n    if (copy_cnt < dest_size)\n        str_dest[copy_cnt] = L'\\0';\n    else\n        str_dest[dest_size - 1] = L'\\0';\n}\n\nvoid CCommon::SetThreadLanguage(Language language)\n{\n    switch (language)\n    {\n    case Language::ENGLISH: SetThreadUILanguage(MAKELANGID(LANG_ENGLISH, SUBLANG_ENGLISH_US)); break;\n    case Language::SIMPLIFIED_CHINESE: SetThreadUILanguage(MAKELANGID(LANG_CHINESE, SUBLANG_CHINESE_SIMPLIFIED)); break;\n    case Language::TRADITIONAL_CHINESE: SetThreadUILanguage(MAKELANGID(LANG_CHINESE, SUBLANG_CHINESE_TRADITIONAL)); break;\n    default: break;\n    }\n}\n\nvoid CCommon::SetColorMode(ColorMode mode)\n{\n    switch (mode)\n    {\n    case ColorMode::Default:\n        CTrafficMonitorApp::self->m_taskbar_data.dft_back_color = 0;\n        CTrafficMonitorApp::self->m_taskbar_data.dft_transparent_color = 0;\n        CTrafficMonitorApp::self->m_taskbar_data.dft_status_bar_color = 0x005A5A5A;\n        CTrafficMonitorApp::self->m_taskbar_data.dft_text_colors = 0x00ffffffU;\n        CTrafficMonitorApp::self->m_cfg_data.m_dft_notify_icon = 0;\n        break;\n    case ColorMode::Light:\n        CTrafficMonitorApp::self->m_taskbar_data.dft_back_color = 0x00D3D2D2;\n        CTrafficMonitorApp::self->m_taskbar_data.dft_transparent_color = 0x00D3D2D2;\n        CTrafficMonitorApp::self->m_taskbar_data.dft_status_bar_color = 0x00A5A5A5;\n        CTrafficMonitorApp::self->m_taskbar_data.dft_text_colors = 0x00000000U;\n        CTrafficMonitorApp::self->m_cfg_data.m_dft_notify_icon = 4;\n        break;\n    default:\n        break;\n    }\n}\n\nvoid CCommon::TransparentColorConvert(COLORREF& transparent_color)\n{\n    if (transparent_color == 0)\n        return;\n    BYTE r = GetRValue(transparent_color);\n    BYTE g = GetGValue(transparent_color);\n    BYTE b = GetBValue(transparent_color);\n    if (r == b)\n    {\n        if (b >= 255)\n            b--;\n        else\n            b++;\n        transparent_color = RGB(r, g, b);\n    }\n}\n\nvoid CCommon::SetDialogFont(CWnd* pDlg, CFont* pFont)\n{\n    if (pDlg->GetSafeHwnd() != NULL)\n    {\n        CWnd* pWndChild;\n        pWndChild = pDlg->GetWindow(GW_CHILD);\n        while (pWndChild)\n        {\n            pWndChild->SetFont(pFont);\n            pWndChild = pWndChild->GetWindow(GW_HWNDNEXT);\n        }\n    }\n}\n\nCString CCommon::GetTextResource(UINT id, int code_type)\n{\n    CString res_str;\n    HRSRC hRes = FindResource(NULL, MAKEINTRESOURCE(id), _T(\"TEXT\"));\n    if (hRes != NULL)\n    {\n        HGLOBAL hglobal = LoadResource(NULL, hRes);\n        if (hglobal != NULL)\n        {\n            if (code_type == 2)\n            {\n                res_str = (const wchar_t*)hglobal;\n            }\n            else\n            {\n                res_str = CCommon::StrToUnicode((const char*)hglobal, (code_type != 0)).c_str();\n            }\n        }\n    }\n    return res_str;\n}\n\nCString CCommon::GetLastCompileTime()\n{\n    CString str_compile_time = GetTextResource(IDR_COMPILE_TIME, 0);\n    str_compile_time.Replace(_T(\"\\r\\n\"), _T(\"\"));\n    str_compile_time.Delete(str_compile_time.GetLength() - 1, 1);\n    return str_compile_time;\n}\n\nHICON CCommon::LoadIconResource(UINT id, int size)\n{\n    return (HICON)LoadImage(AfxGetInstanceHandle(), MAKEINTRESOURCE(id), IMAGE_ICON, size, size, 0);\n}\n\nint CCommon::GetMenuItemPosition(CMenu* pMenu, UINT id)\n{\n    int pos = -1;\n    int item_count = pMenu->GetMenuItemCount();\n    for (int i = 0; i < item_count; i++)\n    {\n        if (pMenu->GetMenuItemID(i) == id)\n        {\n            pos = i;\n            break;\n        }\n    }\n    return pos;\n}\n\nbool CCommon::IsColorSimilar(COLORREF color1, COLORREF color2)\n{\n    const int DIFF{ 24 };\n    return (std::abs(GetRValue(color1) - GetRValue(color2)) < DIFF\n        && std::abs(GetGValue(color1) - GetGValue(color2)) < DIFF\n        && std::abs(GetBValue(color1) - GetBValue(color2)) < DIFF);\n}\n\nint CCommon::CountOneBits(unsigned int value)\n{\n    int count = 0;\n    while (value != 0)\n    {\n        if (value % 2 == 1)\n        {\n            count++;\n        }\n        value = value >> 1;\n    }\n    return count;\n}\n\nvoid CCommon::SetNumberBit(unsigned int& num, int bit, bool value)\n{\n    if (value)\n    {\n        num |= (1 << bit);\n    }\n    else\n    {\n        num &= ~(1 << bit);\n    }\n}\n\nbool CCommon::GetNumberBit(unsigned int num, int bit)\n{\n    return (num & (1 << bit)) != 0;\n}\n"
  },
  {
    "path": "TrafficMonitor/Common.h",
    "content": "﻿#pragma once\n#include \"CommonData.h\"\n#include \"CVariant.h\"\n#include <set>\n#include <functional>\n\nclass CCommon\n{\npublic:\n    CCommon();\n    ~CCommon();\n    //将const char*字符串转换成宽字符字符串\n    static wstring StrToUnicode(const char* str, bool utf8 = false);\n\n    static string UnicodeToStr(const wchar_t* wstr, bool utf8 = false);\n\n    template<class T>\n    static void StringNormalize(T& str)\n    {\n        if (str.empty()) return;\n\n        int size = str.size();  //字符串的长度\n        if (size < 0) return;\n        int index1 = 0;     //字符串中第1个不是空格或控制字符的位置\n        int index2 = size - 1;  //字符串中最后一个不是空格或控制字符的位置\n        while (index1 < size && str[index1] >= 0 && str[index1] <= 32)\n            index1++;\n        while (index2 >= 0 && str[index2] >= 0 && str[index2] <= 32)\n            index2--;\n        if (index1 > index2)    //如果index1 > index2，说明字符串全是空格或控制字符\n            str.clear();\n        else if (index1 == 0 && index2 == size - 1) //如果index1和index2的值分别为0和size - 1，说明字符串前后没有空格或控制字符，直接返回\n            return;\n        else\n            str = str.substr(index1, index2 - index1 + 1);\n    }\n\n    //将一个字符串分割成若干个字符（模板类型只能为string或wstring）\n    //str: 原始字符串\n    //div_ch: 用于分割的字符\n    //result: 接收分割后的结果\n    template<class T>\n    static void StringSplit(const T& str, wchar_t div_ch, vector<T>& results, bool skip_empty = true, bool trim = true)\n    {\n        results.clear();\n        size_t split_index = -1;\n        size_t last_split_index = -1;\n        while (true)\n        {\n            split_index = str.find(div_ch, split_index + 1);\n            T split_str = str.substr(last_split_index + 1, split_index - last_split_index - 1);\n            if (trim)\n                StringNormalize(split_str);\n            if (!split_str.empty() || !skip_empty)\n                results.push_back(split_str);\n            if (split_index == wstring::npos)\n                break;\n            last_split_index = split_index;\n        }\n    }\n\n    template<class T>\n    static void StringSplit(const T& str, const T& div_str, vector<T>& results, bool skip_empty = true, bool trim = true)\n    {\n        results.clear();\n        size_t split_index = 0 - div_str.size();\n        size_t last_split_index = 0 - div_str.size();\n        while (true)\n        {\n            split_index = str.find(div_str, split_index + div_str.size());\n            T split_str = str.substr(last_split_index + div_str.size(), split_index - last_split_index - div_str.size());\n            if (trim)\n                StringNormalize(split_str);\n            if (!split_str.empty() || !skip_empty)\n                results.push_back(split_str);\n            if (split_index == wstring::npos)\n                break;\n            last_split_index = split_index;\n        }\n    }\n\n\n    template<class T>\n    static bool StringTransform(T& str, bool upper)\n    {\n        if (str.empty()) return false;\n        if (upper)\n        {\n            for (auto& ch : str)\n            {\n                {\n                    if (ch >= 'a' && ch <= 'z')\n                        ch -= 32;\n                }\n            }\n        }\n        else\n        {\n            for (auto& ch : str)\n            {\n                if (ch >= 'A' && ch <= 'Z')\n                    ch += 32;\n            }\n        }\n        return true;\n    }\n\n    //读取文件内容\n    static bool GetFileContent(const wchar_t* file_path, string& contents_buff, bool binary = true);\n\n    //读取文件内容\n    //file_path: 文件的路径\n    //length: 文件的长度\n    //binary: 是否以进制方式读取\n    //返回值: 读取成功返回读取到的文件内容的const char类型的指针，在使用完毕后这个指针需要自行使用delete释放。读取失败返回nullptr\n    static const char* GetFileContent(const wchar_t* file_path, size_t& length, bool binary = true);\n\n    /*根据数据的大小转换成以KB、MB、GB为单位的字符串\n    size：数据的字节数\n    返回值：转换后的字符串\n    */\n    static CString DataSizeToString(unsigned long long size, const PublicSettingData& cfg);\n\n    /*根据数据的大小转换成以KB、MB、GB为单位的字符串\n    size：数据的字节数\n    with_space：数值和单位是否使用空格分隔\n    返回值：转换后的字符串\n    */\n    static CString DataSizeToString(unsigned long long size, bool with_space = true);\n\n    //将温度信息转换成字符串\n    static CString TemperatureToString(float temperature, const PublicSettingData& cfg);\n\n    //将使用率转换成字符串\n    static CString UsageToString(int usage, const PublicSettingData& cfg);\n    static CString FreqToString(float usage, const PublicSettingData& cfg);\n    //static CString KBytesToString(unsigned int kb_size);\n    static CString KBytesToString(unsigned __int64 kb_size);\n\n    //返回两个FILETIME结构的时间差\n    static __int64 CompareFileTime2(FILETIME time1, FILETIME time2);\n\n    //将一个日志信息str_text写入到file_path文件中\n    static void WriteLog(const char* str_text, LPCTSTR file_path);\n    static void WriteLog(const wchar_t* str_text, LPCTSTR file_path);\n\n    /*\n    函数功能：对指定文件在指定的目录下创建其快捷方式\n    函数参数：\n    lpszLnkFileDir  指定目录，不能为NULL。\n    lpszFileName    指定文件，为NULL表示当前进程的EXE文件。\n    lpszLnkFileName 快捷方式名称，为NULL表示EXE文件名。\n    wHotkey         为0表示不设置快捷键\n    pszDescription  备注\n    iShowCmd        运行方式，默认为常规窗口\n    */\n    static BOOL CreateFileShortcut(LPCTSTR lpszLnkFileDir, LPCTSTR lpszFileName = NULL, LPCTSTR lpszLnkFileName = NULL, LPCTSTR lpszWorkDir = NULL, WORD wHotkey = 0, LPCTSTR lpszDescription = NULL, int iShowCmd = SW_SHOWNORMAL);\n\n    //获取开始菜单“所有程序”中的“启动”目录的路径\n    static wstring GetStartUpPath();\n\n    //获取path路径下的文件或文件夹，并将文件或文件夹名称保存在files容器中。\n    static void GetFiles(const wchar_t* path, vector<wstring>& files);\n\n    //获取path路径下的文件或文件夹，每次遍历时调用函数对数func\n    //path: 查找的路径\n    //func: 可以是一个函数对象或lambda表达式，参数是遍历到的文件或文件夹名\n    static void GetFiles(const wchar_t* path, std::function<void(const wstring&)> func);\n\n    //判断一个文件是否存在\n    static bool FileExist(LPCTSTR file_name);\n\n    //判断是否是文件夹\n    static bool IsFolder(const wstring& path);\n\n    static bool MoveAFile(LPCTSTR exist_file, LPCTSTR new_file);\n\n    //计算两个SYSTEMTIME结构时间的差（a-b，只保留时、分、秒）\n    static SYSTEMTIME CompareSystemTime(SYSTEMTIME a, SYSTEMTIME b);\n\n    //获取当前程序的目录\n    static wstring GetModuleDir();\n\n    //获取system32文件夹的路径\n    static wstring GetSystemDir();\n\n    //获取临时文件夹的路径\n    static wstring GetTemplateDir();\n\n    //获取Appdata/Local/TrafficMonitor的目录，如果不存在，则会自动创建\n    static wstring GetAppDataConfigDir();\n\n    //在指定位置绘制文本\n    static void DrawWindowText(CDC* pDC, CRect rect, LPCTSTR lpszString, COLORREF color, COLORREF back_color);\n\n    ////设置绘图的剪辑区域\n    //static void SetDrawArea(CDC* pDC, CRect rect);\n\n    /* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * *\n    * 函数名称：IsForegroundFullscreen\n    * 功能说明：判断当前正在与用户交互的前台窗口是否是全屏的。\n    * 参数说明：无\n    * 返回说明：true：是。\n    false：否。\n    * 线程安全：是\n    * 调用样例：IsForegroundFullscreen ()，表示判断当前正在与用户交互的前台窗口是否是全屏的。\n    * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */\n    static bool IsForegroundFullscreen();\n\n    //将一个字符串保存到剪贴板\n    static bool CopyStringToClipboard(const wstring& str);\n\n    static wstring GetJsonValueSimple(const wstring& json_str, const wstring& name);\n\n    //获取URL的内容\n    static bool GetURL(const wstring& url, wstring& result, bool utf8 = false, const wstring& user_agent = wstring());\n\n    //获取外网IP地址和IP归属地\n    static void GetInternetIp(wstring& ip_address, wstring& ip_location, bool global);\n\n    static void GetInternetIp2(wstring& ip_address, wstring& ip_location, bool ipv6);\n\n    static void SetRect(CRect& rect, int x, int y, int width, int height);\n\n    //从资源文件载入字符串。其中，front_str、back_str为载入字符串时需要在前面或后面添加的字符串\n    static CString LoadText(UINT id, LPCTSTR back_str = nullptr);\n    static CString LoadText(LPCTSTR front_str, UINT id, LPCTSTR back_str = nullptr);\n\n    //安全的格式化字符串，将format_str中形如<%序号%>的字符串替换成初始化列表paras中的元素，元素支持int/double/LPCTSTR/CString格式，序号从1开始\n    static CString StringFormat(LPCTSTR format_str, const std::initializer_list<CVariant>& paras);\n\n    //从资源文件中载入字符串，并将资源字符串中形如<%序号%>的字符串替换成可变参数列表中的参数\n    static CString LoadTextFormat(UINT id, const std::initializer_list<CVariant>& paras);\n\n    //将int类型转换成字符串\n    //n：要转换的数值\n    //thousand_separation：是否要每隔3位数使用逗号分隔\n    //is_unsigned：数值是否是无符号的\n    static CString IntToString(__int64 n, bool thousand_separation = false, bool is_unsigned = false);\n\n    //删除字体名称后面的Bold、Light等字符串，并根据这些字符串设置字体粗细\n    static void NormalizeFont(LOGFONT& font);\n\n    //安全的字符串复制函数\n    static void WStringCopy(wchar_t* str_dest, int dest_size, const wchar_t* str_source, int source_size = INT_MAX);\n\n    /// <summary>\n    /// 字符串相似度算法-编辑距离法\n    /// </summary>\n    /// <returns>返回的值为0~1，越大相似度越高</returns>\n    template<class T>\n    static double StringSimilarDegree_LD(const T& srcString, const T& matchString)\n    {\n        int n = srcString.size();\n        int m = matchString.size();\n        //int[, ] d = new int[n + 1, m + 1]; // matrix\n        vector<vector<int>> d(n + 1, vector<int>(m + 1));\n        int cost; // cost\n                  // Step 1（如果其中一个字符串长度为0，则相似度为1）？\n                  //if (n == 0) return (double)m / max(srcString.size(), matchString.size());\n                  //if (m == 0) return (double)n / max(srcString.size(), matchString.size());\n        if (n == 0 || m == 0) return 0.0;   //如果其中一个字符串长度为0，则相似度为0\n                                            // Step 2\n        for (int i = 0; i <= n; d[i][0] = i++);\n        for (int j = 0; j <= m; d[0][j] = j++);\n        // Step 3\n        for (int i = 1; i <= n; i++)\n        {\n            //Step 4\n            for (int j = 1; j <= m; j++)\n            {\n                // Step 5\n                cost = (matchString.substr(j - 1, 1) == srcString.substr(i - 1, 1) ? 0 : 1);\n                // Step 6\n                d[i][j] = min(min(d[i - 1][j] + 1, d[i][j - 1] + 1), d[i - 1][j - 1] + cost);\n            }\n        }\n\n        // Step 7\n        double ds = 1 - (double)d[n][m] / max(srcString.size(), matchString.size());\n\n        return ds;\n    }\n\n    //设置线程语言\n    static void SetThreadLanguage(Language language);\n\n    //设置颜色模式\n    static void SetColorMode(ColorMode mode);\n\n    //经过测试发现，似乎当透明色的R和B值相等时，会出现右击任务栏窗口时无法弹出右键菜单，而是弹出系统任务栏右键菜单的问题\n    //为了解决这个问题，需要将颜色值进行转换\n    //此函数的作用是判断一个颜色的R和B值是否相等，如果是则将颜色的B值加1（如果B==255，则减1）\n    static void TransparentColorConvert(COLORREF& transparent_color);\n\n    static void SetDialogFont(CWnd* pDlg, CFont* pFont);\n\n    //从资源加载自定义文本资源。id：资源的ID，code_type：文本的编码格式：0:ANSI, 1:UTF8, 2:UTF16\n    static CString GetTextResource(UINT id, int code_type);\n\n    //从资源文件读取上次编译时间\n    static CString GetLastCompileTime();\n\n    //从资源加载一个图标\n    static HICON LoadIconResource(UINT id, int size);\n\n    //获取一个菜单项的序号\n    static int GetMenuItemPosition(CMenu* pMenu, UINT id);\n\n    static bool IsColorSimilar(COLORREF color1, COLORREF color2);\n\n    //计算二进制中1的个数\n    static int CountOneBits(unsigned int value);\n\n    //设置一个数字的某个bit位\n    static void SetNumberBit(unsigned int& num, int bit, bool value);\n\n    //获取一个数字的某个bit位\n    static bool GetNumberBit(unsigned int num, int bit);\n\n    //在不排序的情况下删除vector中的重复元素\n    template<class T>\n    static void RemoveVectorDuplicateItem(vector<T>& vec)\n    {\n        std::set<int> si;\n        for (auto it = vec.begin(); it != vec.end();)\n        {\n            if (si.count(*it) == 0)//这里判断当前元素是否已经出现过，若没有出现过则将该元素保留，并做标记\n            {\n                si.insert(*it);\n                ++it;\n            }\n            else//如果前面已经出现过，则删除当前元素\n            {\n                it = vec.erase(it);//这里删除it指向的元素\n            }\n        }\n    }\n\n};\n"
  },
  {
    "path": "TrafficMonitor/CommonData.cpp",
    "content": "﻿#include \"stdafx.h\"\n#include \"CommonData.h\"\n#include \"Common.h\"\n#include \"CalendarHelper.h\"\n#include \"TrafficMonitor.h\"\n\n///////////////////////////////////////////////////////////////////////////////////\nint Date::week() const\n{\n    //计算当前是一年的第几天\n    int days{};\n    for (int i{ 1 }; i < month; i++)\n    {\n        days += CCalendarHelper::DaysInMonth(year, i);\n    }\n    days += day;\n    //计算这一年的1月1日是星期几\n    int week_day = CCalendarHelper::CaculateWeekDay(year, 1, 1);\n    if (theApp.m_cfg_data.m_sunday_first)\n    {\n        days += (week_day - 1);\n    }\n    else\n    {\n        days += (week_day - 2);\n    }\n    return days / 7 + 1;\n}\n\nbool Date::DateGreater(const Date& a, const Date& b)\n{\n    if (a.year != b.year)\n        return a.year > b.year;\n    else if (a.month != b.month)\n        return a.month > b.month;\n    else if (a.day != b.day)\n        return a.day > b.day;\n    else\n        return false;\n}\n\nbool Date::DateEqual(const Date& a, const Date& b)\n{\n    return a.year == b.year && a.month == b.month && a.day == b.day;\n}\n\n\n///////////////////////////////////////////////////////////////////////////////////\n//HistoryTraffic\nunsigned __int64 HistoryTraffic::kBytes() const\n{\n    return up_kBytes + down_kBytes;\n}\n\n\n///////////////////////////////////////////////////////////////////////////////////\nwstring& DispStrings::Get(CommonDisplayItem item)\n{\n    return map_str[item];\n}\n\nconst std::map<CommonDisplayItem, wstring>& DispStrings::GetAllItems() const\n{\n    return map_str;\n}\n\nvoid DispStrings::operator=(const DispStrings& disp_str)\n{\n    map_str = disp_str.map_str;\n    //如果赋值的字符串是定义的无效字符串，则不赋值\n    for (auto& iter = map_str.begin(); iter != map_str.end(); ++iter)\n    {\n        if (iter->second == NONE_STR)\n            iter->second.clear();\n    }\n}\n\nbool DispStrings::IsInvalid() const\n{\n    for (auto& iter = map_str.begin(); iter != map_str.end(); ++iter)\n    {\n        if (iter->second == NONE_STR)\n            return true;\n    }\n    return false;\n}\n\nvoid DispStrings::Load(const std::wstring& plugin_id, const std::wstring& disp_str)\n{\n    auto plugin = theApp.m_plugins.GetItemById(plugin_id);\n    if (plugin != nullptr)\n    {\n        map_str[plugin] = disp_str;\n    }\n}\n\n///////////////////////////////////////////////////////////////////////////////////\nbool StringSet::Contains(const std::wstring& str) const\n{\n    return string_set.count(str) != 0;\n}\n\nvoid StringSet::SetStrContained(const std::wstring& str, bool contained)\n{\n    if (contained)\n        string_set.insert(str);\n    else\n        string_set.erase(str);\n}\n\nvoid StringSet::FromString(const std::wstring& str)\n{\n    std::vector<std::wstring> item_vect;\n    CCommon::StringSplit(str, L',', item_vect);\n    string_set.clear();\n    for (const auto& i : item_vect)\n        string_set.insert(i);\n}\n\nstd::wstring StringSet::ToString() const\n{\n    std::vector<std::wstring> item_vect;\n    for (const auto& i : string_set)\n        item_vect.push_back(i);\n    std::wstring item_str;\n    for (const auto& i : item_vect)\n    {\n        item_str += i;\n        item_str += L',';\n    }\n    if (!item_str.empty())\n        item_str.pop_back();\n    return item_str;\n}\n\nvoid StringSet::FromVector(const std::vector<std::wstring>& vec)\n{\n    string_set.clear();\n    for (const auto& str : vec)\n        string_set.insert(str);\n}\n\nstd::vector<std::wstring> StringSet::ToVector() const\n{\n    std::vector<std::wstring> vec;\n    for (const auto& str : string_set)\n        vec.push_back(str);\n    return vec;\n}\n\nstd::set<std::wstring>& StringSet::data()\n{\n    return string_set;\n}\n\nvoid TaskBarSettingData::ValidItemSpace()\n{\n    if (item_space < 0)\n        item_space = 0;\n    if (item_space > 32)\n        item_space = 32;\n}\n\nunsigned __int64 TaskBarSettingData::GetNetspeedFigureMaxValueInBytes() const\n{\n    if (netspeed_figure_max_value_unit == 0)        //单位为KB\n        return static_cast<unsigned __int64>(netspeed_figure_max_value) * 1024;\n    else\n        return static_cast<unsigned __int64>(netspeed_figure_max_value) * 1024 * 1024;\n}\n"
  },
  {
    "path": "TrafficMonitor/CommonData.h",
    "content": "﻿//此文件包含全局结构体、枚举类型的定义\n#pragma once\n#include \"stdafx.h\"\n#include \"TaskbarItemOrderHelper.h\"\n\nstruct Date\n{\n    int year{};\n    int month{};\n    int day{};\n\n    int week() const;       //该日期是一年的第几周\n\n    //比较两个HistoryTraffic对象的日期，如果a的时间大于b，则返回true\n    static bool DateGreater(const Date& a, const Date& b);\n\n    //判断两个HistoryTraffic对象的日期是否相等\n    static bool DateEqual(const Date& a, const Date& b);\n\n};\n\n//储存某一天的历史流量\nstruct HistoryTraffic : public Date\n{\n    //当天使用的流量（以KB为单位）\n    unsigned __int64 up_kBytes{};\n    unsigned __int64 down_kBytes{};\n    bool mixed{ true };     //如果不区分上传和下载流量，则为true\n\n    unsigned __int64 kBytes() const;\n\n};\n\n//历史流量统计中用于指示不同范围内的流量的颜色\n#define TRAFFIC_COLOR_BLUE RGB(0, 183, 238)\n#define TRAFFIC_COLOR_GREEN RGB(128, 194, 105)\n#define TRAFFIC_COLOR_YELLOE RGB(255, 216, 58)\n#define TRAFFIC_COLOR_RED RGB(255, 95, 74)\n#define TRAFFIC_COLOR_DARK_RED RGB(166, 19, 0)\n\n//网速单位\nenum class SpeedUnit\n{\n    AUTO,       //自动\n    KBPS,       //KB/s\n    MBPS        //MB/s\n};\n\n\n//硬件监控的项目\nenum HardwareItem\n{\n    HI_CPU = 1 << 0,        //CPU\n    HI_GPU = 1 << 1,        //显卡\n    HI_HDD = 1 << 2,        //硬盘\n    HI_MBD = 1 << 3         //主板\n};\n\n#define DEF_CH L'\\\"'        //写入和读取ini文件字符串时，在字符串前后添加的字符\n#define NONE_STR L\"@@@\"     //用于指定一个无效字符串\nstruct DispStrings      //显示的文本\n{\nprivate:\n    std::map<CommonDisplayItem, wstring> map_str;\n\npublic:\n    //获取一个显示的文本\n    wstring& Get(CommonDisplayItem item);\n\n    const std::map<CommonDisplayItem, wstring>& GetAllItems() const;\n\n    void operator=(const DispStrings& disp_str);     //重载赋值运算符\n\n    //载入一个插件项目的显示文本\n    void Load(const std::wstring& plugin_id, const std::wstring& disp_str);\n\n    //是否无效\n    bool IsInvalid() const;\n};\n\n//鼠标双击窗口的动作\nenum class DoubleClickAction\n{\n    CONNECTION_INFO,    //连接详情\n    HISTORY_TRAFFIC,    //历史流量统计\n    SHOW_MORE_INFO,     //显示更多信息\n    OPTIONS,            //选项设置\n    TASK_MANAGER,       //任务管理器\n    SEPCIFIC_APP,       //指定应用程序\n    CHANGE_SKIN,        //更换皮肤\n    NONE                //不执行任何动作\n};\n\n//语言\nenum class Language\n{\n    FOLLOWING_SYSTEM,       //跟随系统\n    ENGLISH,                //英语\n    SIMPLIFIED_CHINESE,     //简体中文\n    TRADITIONAL_CHINESE     //繁体中文\n};\n\n//颜色模式\nenum class ColorMode\n{\n    Default, //默认颜色\n    Light   //浅色\n};\n\n//将字号转成LOGFONT结构中的lfHeight\ninline int FontSizeToLfHeight(int font_size, int dpi = 0)\n{\n    if (dpi == 0)\n    {\n        HDC hDC = ::GetDC(HWND_DESKTOP);\n        dpi = GetDeviceCaps(hDC, LOGPIXELSY);\n        ::ReleaseDC(HWND_DESKTOP, hDC);\n    }\n    int lfHeight = -MulDiv(font_size, dpi, 72);\n    return lfHeight;\n}\n\n//字体\nstruct FontInfo\n{\n    CString name;   //字体名称\n    int size;       //字体大小\n    bool bold;          //粗体\n    bool italic;        //斜体\n    bool underline;     //下划线\n    bool strike_out;    //删除线\n\n    //创建一个CFont对象\n    void Create(CFont& font, int dpi = 0)\n    {\n        font.CreateFont(\n            FontSizeToLfHeight(size, dpi), // nHeight\n            0, // nWidth\n            0, // nEscapement\n            0, // nOrientation\n            (bold ? FW_BOLD : FW_NORMAL), // nWeight\n            italic, // bItalic\n            underline, // bUnderline\n            strike_out, // cStrikeOut\n            DEFAULT_CHARSET, // nCharSet\n            OUT_DEFAULT_PRECIS, // nOutPrecision\n            CLIP_DEFAULT_PRECIS, // nClipPrecision\n            DEFAULT_QUALITY, // nQuality\n            DEFAULT_PITCH | FF_SWISS, // nPitchAndFamily\n            name);\n    }\n};\n\n//历史流量统计列表视图中显示模式\nenum class HistoryTrafficViewType\n{\n    HV_DAY,         //日视图\n    HV_WEEK,        //周视图\n    HV_MONTH,          //月视图\n    HV_QUARTER,     //季视图\n    HV_YEAR            //年视图\n};\n\nstruct StringSet\n{\npublic:\n    bool Contains(const std::wstring& str) const;\n    void SetStrContained(const std::wstring& str, bool contained);\n    void FromString(const std::wstring& str);\n    std::wstring ToString() const;\n    void FromVector(const std::vector<std::wstring>& vec);\n    std::vector<std::wstring> ToVector() const;\n    std::set<std::wstring>& data();\nprivate:\n    std::set<std::wstring> string_set;\n};\n\n//选项设置数据\nstruct MainConfigData\n{\n    int m_transparency{ 100 };          //窗口透明度\n    bool m_show_more_info{ false };     //显示更多信息\n    bool m_show_task_bar_wnd{ false };  //显示任务栏窗口\n    bool m_hide_main_window;            //隐藏主窗口\n    //bool m_tbar_show_cpu_memory;      //任务栏窗口显示CPU和内存利用率\n\n    int m_position_x;   //窗口位置的x坐标\n    int m_position_y;   //窗口位置的y坐标\n\n    bool m_auto_select{ false };    //自动选择连接\n    bool m_select_all{ false };     //统计所有连接的网速\n    string m_connection_name;      //当前选择网络的名称\n\n    wstring m_skin_name;            //选择的皮肤的名称\n    int m_dft_notify_icon = 0;      //默认的通知图标(用于区分win10的深色和浅色模式)\n    int m_notify_icon_selected{};   //要显示的通知区图标\n    bool m_notify_icon_auto_adapt{ false }; //通知区图标是否自动适应Win10深浅色模式\n\n    //bool m_show_internet_ip{ false };     //是否在“连接详情”对话框中显示外网IP地址\n    bool m_use_log_scale{ false };          //“历史流量统计”对话框中绘制表示历史流量数值的矩形时是否使用对数比例\n    HistoryTrafficViewType m_view_type{};\n    bool m_sunday_first{ true };            //是否将周日作为一周的第一天\n    StringSet plugin_disabled;      //已禁用的插件\n\n    int taskbar_left_space_win11{};         //Windows11下，任务栏窗口显示在左侧时的边距\n};\n\n//内存显示方式\nenum class MemoryDisplay\n{\n    USAGE_PERCENTAGE,       //已使用百分比\n    MEMORY_USED,            //内存已使用\n    MEMORY_AVAILABLE        //内存可用\n};\n\n//选项设置中“主窗口设置”和“任务栏窗口设置”中公共的数据（不使用此结构体创建对象）\nstruct PublicSettingData\n{\n    bool specify_each_item_color{ false };      //是否指定每个项目的颜色\n    FontInfo font;          //字体\n    DispStrings disp_str;   //显示的文本\n    bool speed_short_mode{ false };     //网速显示简洁模式（减少小数点的位数，单位不显示“B”）\n    bool separate_value_unit_with_space{ true };    //网速数值和单位用空格分隔\n    bool show_tool_tip{ true };         //显示鼠标提示\n    MemoryDisplay memory_display{ MemoryDisplay::USAGE_PERCENTAGE };    //内存显示方式\n\n    bool unit_byte{ true };             //使用字节(B)而不是比特(b)为单位\n    SpeedUnit speed_unit;       //网速的单位\n    bool hide_unit;         //隐藏单位\n    bool hide_percent;      //隐藏百分号\n    DoubleClickAction double_click_action;      //鼠标双击动作\n    wstring double_click_exe;   //鼠标双击动作为打开指定应用程序时，打开的程序路径\n};\n\n//#define MAIN_WND_COLOR_NUM 9      //主窗口颜色数量\n//选项设置中“主窗口设置”的数据\nstruct MainWndSettingData : public PublicSettingData\n{\n    std::map<CommonDisplayItem, COLORREF> text_colors{};    //方字的颜色\n    bool swap_up_down{ false };     //交换上传和下载显示的位置\n    bool hide_main_wnd_when_fullscreen;     //有程序全屏运行时隐藏悬浮窗\n    bool m_always_on_top{ false };      //窗口置顶\n    bool m_lock_window_pos{ false };    //锁定窗口位置\n    bool m_mouse_penetrate{ false };    //鼠标穿透\n    bool m_alow_out_of_border{ false };     //是否允许悬浮窗超出屏幕边界\n\n};\n\n//#define TASKBAR_COLOR_NUM 18      //任务栏窗口颜色数量\n\nstruct TaskbarItemColor //任务栏窗口每一项的颜色\n{\n    COLORREF label{};   //标签颜色\n    COLORREF value{};   //数值颜色\n\n    bool operator==(const TaskbarItemColor& item) const\n    {\n        return label == item.label && value == item.value;\n    }\n};\n\n//选项设置中“任务栏窗口设置”的数据\nstruct TaskBarSettingData : public PublicSettingData\n{\n    COLORREF  back_color{ RGB(0, 0, 0) };   //背景颜色\n    COLORREF transparent_color{ RGB(0, 0, 0) };     //透明色\n    COLORREF status_bar_color{ RGB(0, 0, 0) };      // CPU/内存 状态条颜色\n    std::map<CommonDisplayItem, TaskbarItemColor> text_colors{};    //文字的颜色\n    int dft_back_color = 0;                         //默认背景颜色\n    int dft_transparent_color = 0;                  //默认透明色\n    int dft_status_bar_color = 0x005A5A5A;          //默认CPU/内存 状态条颜色\n    int dft_text_colors = 0x00ffffffU;              //默认文字颜色\n\n    bool auto_adapt_light_theme{ true };            //是否自动适应浅色主题\n    int dark_default_style{ 0 };                    //深色主题时使用的预设方案\n    int light_default_style{ -1 };                  //浅色主题时使用的预设方案\n    bool auto_set_background_color{ false };        //根据任务栏颜色自动设置背景色\n    bool auto_save_taskbar_color_settings_to_preset{};    //当启用“自动适应Windows10深色/浅色主题”时，是否在颜色设置有更改时自动将当前颜色设置保存到对应的预设\n\n    CTaskbarItemOrderHelper item_order;\n    unsigned int m_tbar_display_item{ TDI_UP | TDI_DOWN };      //任务栏窗口显示的项目\n    StringSet plugin_display_item;                  //任务窗口显示的插件项目\n\n    bool value_right_align{ false };    //数值是否右对齐\n    int digits_number{ 4 };             //数据位数\n    bool horizontal_arrange{ true };    //水平排列\n    bool show_status_bar{ true };       //显示 CPU/内存的状态条\n    bool tbar_wnd_on_left{ false };     //如果为true，则任务栏窗口显示在任务栏的左侧（或上方）\n    bool tbar_wnd_snap{ false };     \t//如果为true，则在Win11中任务栏窗口贴靠中间任务栏，否则靠近边缘\n    bool cm_graph_type{ false };        //如果为false，默认原样式，柱状图显示占用率，如为true，滚动显示占用率\n    bool show_graph_dashed_box{ true }; //显示占用图虚线框\n    int item_space{};                   //任务栏项目间距\n    void ValidItemSpace();\n\n    bool show_netspeed_figure{ false };     //是否显示网速占用图\n    int netspeed_figure_max_value;          //网速占用图的最大值\n    int netspeed_figure_max_value_unit{};   //网速占用图最大值的单位（0: KB, 1: MB）\n    unsigned __int64 GetNetspeedFigureMaxValueInBytes() const;  //获取网速占用图的最大值（以字节为单位）\n\n};\n\n//选项设置中“常规设置”的数据\nstruct GeneralSettingData\n{\n    bool check_update_when_start{ true };\n    int update_source{};                    //更新源。0: GitHub; 1: Gitee\n    bool auto_run{ false };\n    bool allow_skin_cover_font{ true };\n    bool allow_skin_cover_text{ true };\n    bool show_notify_icon{ true };    //显示通知区域图标\n//通知消息\n    bool traffic_tip_enable{ false };       //是否启用流量超出时提示\n    int traffic_tip_value;                  //要提示的流量临界值\n    int traffic_tip_unit{};                 //要提示的流量值的单位（0: MB, 1: GB）\n\n    struct NotifyTipSettings        //超过某个值时弹出提示的设置\n    {\n        bool enable;                //是否启用提示\n        int tip_value;              //要弹出提示的临界值\n    };\n    NotifyTipSettings memory_usage_tip;     //用内存使用率超出提示\n    NotifyTipSettings cpu_temp_tip;         //CPU温度超出提示\n    NotifyTipSettings gpu_temp_tip;         //显卡温度超出提示\n    NotifyTipSettings hdd_temp_tip;         //硬盘温度超出提示\n    NotifyTipSettings mainboard_temp_tip;   //主板温度超出提示\n\n\n    //语言\n    Language language;\n\n    bool show_all_interface{ true };\n    bool m_get_cpu_usage_by_cpu_times{ true };  //获取CPU利用率的方式，如果为true则是使用GetSystemTimes，否则使用Pdh（性能计数器）\n\n    bool portable_mode{ false };        //便携模式，如果为true，则程序所有数据都保存到exe所在目录下，否则保存到Appdata\\Romaing目录下\n    int monitor_time_span{ 1000 };    //监控的时间间隔\n\n    std::wstring hard_disk_name;        //要监控的硬盘名称\n    std::wstring cpu_core_name;         //要监控的CPU核心的名称\n\n    unsigned int hardware_monitor_item{};   //要监控哪些硬件\n    bool IsHardwareEnable(HardwareItem item_type) const\n    {\n        return hardware_monitor_item & item_type;\n    }\n    void SetHardwareEnable(HardwareItem item_type, bool enable)\n    {\n        if (enable)\n            hardware_monitor_item |= item_type;\n        else\n            hardware_monitor_item &= ~item_type;\n    }\n\n    StringSet connections_hide;     //用于保存哪些网络要从“选择网络连接”子菜单项中隐藏\n};\n\n//定义监控时间间隔有效的最大值和最小值\n#define MONITOR_TIME_SPAN_MIN 200\n#define MONITOR_TIME_SPAN_MAX 2000\n\nenum class Alignment\n{\n    LEFT,       //左对齐\n    RIGHT,      //右对齐\n    CENTER,     //居中\n    SIDE        //两端对齐\n};\n\n//通过构造函数传递一个bool变量的引用，在构造时将其置为true，析构时置为false\nclass CFlagLocker\n{\npublic:\n    CFlagLocker(bool& flag)\n        : m_flag(flag)\n    {\n        m_flag = true;\n    }\n\n    ~CFlagLocker()\n    {\n        m_flag = false;\n    }\n\nprivate:\n    bool& m_flag;\n};\n"
  },
  {
    "path": "TrafficMonitor/DisplayTextSettingDlg.cpp",
    "content": "﻿// DisplayTextSettingDlg.cpp: 实现文件\n//\n\n#include \"stdafx.h\"\n#include \"TrafficMonitor.h\"\n#include \"DisplayTextSettingDlg.h\"\n\n\n// CDisplayTextSettingDlg 对话框\n\nIMPLEMENT_DYNAMIC(CDisplayTextSettingDlg, CBaseDialog)\n\nCDisplayTextSettingDlg::CDisplayTextSettingDlg(DispStrings& display_texts, bool main_window_text, CWnd* pParent /*=nullptr*/)\n\t: CBaseDialog(IDD_DISPLAY_TEXT_SETTINGS_DIALOG, pParent), m_display_texts(display_texts), m_main_window_text(main_window_text)\n{\n\n}\n\nCDisplayTextSettingDlg::~CDisplayTextSettingDlg()\n{\n}\n\nvoid CDisplayTextSettingDlg::DoDataExchange(CDataExchange* pDX)\n{\n    CBaseDialog::DoDataExchange(pDX);\n    DDX_Control(pDX, IDC_LIST1, m_list_ctrl);\n}\n\n\nCString CDisplayTextSettingDlg::GetDialogName() const\n{\n    return _T(\"DisplayTextSettingDlg\");\n}\n\nBEGIN_MESSAGE_MAP(CDisplayTextSettingDlg, CBaseDialog)\n    ON_BN_CLICKED(IDC_RESTORE_DEFAULT_BUTTON, &CDisplayTextSettingDlg::OnBnClickedRestoreDefaultButton)\nEND_MESSAGE_MAP()\n\n\n// CDisplayTextSettingDlg 消息处理程序\n\n\nBOOL CDisplayTextSettingDlg::OnInitDialog()\n{\n    CBaseDialog::OnInitDialog();\n\n    // TODO:  在此添加额外的初始化\n\n    SetIcon(theApp.GetMenuIcon(IDI_ITEM), FALSE);\t\t// 设置小图标\n\n    //初始化列表控件\n    CRect rect;\n    m_list_ctrl.GetClientRect(rect);\n    m_list_ctrl.SetExtendedStyle(LVS_EX_FULLROWSELECT | LVS_EX_LABELTIP);\n    int width0, width1;\n    width0 = rect.Width() / 2;\n    width1 = rect.Width() - width0 - theApp.DPI(20) - 1;\n    m_list_ctrl.InsertColumn(0, CCommon::LoadText(IDS_ITEM), LVCFMT_LEFT, width0);\t\t//插入第0列\n    m_list_ctrl.InsertColumn(1, CCommon::LoadText(IDS_VALUE), LVCFMT_LEFT, width1);\t\t//插入第1列\n\n    //向列表中插入行\n    for (auto iter = m_display_texts.GetAllItems().begin(); iter != m_display_texts.GetAllItems().end(); ++iter)\n    {\n        CString item_name = iter->first.GetItemName();\n        if (!item_name.IsEmpty())\n        {\n            int index = m_list_ctrl.GetItemCount();\n            m_list_ctrl.InsertItem(index, item_name);\n            m_list_ctrl.SetItemText(index, 1, iter->second.c_str());\n            m_list_ctrl.SetItemData(index, (DWORD_PTR)&(iter->first));\n        }\n    }\n\n    m_list_ctrl.SetEditColMethod(CListCtrlEx::EC_SPECIFIED);        //设置列表可编辑\n    m_list_ctrl.SetEditableCol({ 1 });                              //设置可编辑的列\n\n    return TRUE;  // return TRUE unless you set the focus to a control\n                  // 异常: OCX 属性页应返回 FALSE\n}\n\nCommonDisplayItem CDisplayTextSettingDlg::GetDisplayItem(int row)\n{\n    CommonDisplayItem* item = (CommonDisplayItem*)m_list_ctrl.GetItemData(row);\n    return *item;\n}\n\nvoid CDisplayTextSettingDlg::OnOK()\n{\n    // TODO: 在此添加专用代码和/或调用基类\n\n    int item_count = m_list_ctrl.GetItemCount();\n    for (int i{}; i < item_count; i++)\n    {\n        CommonDisplayItem display_item = GetDisplayItem(i);\n        m_display_texts.Get(display_item) = m_list_ctrl.GetItemText(i, 1).GetString();\n    }\n\n    CBaseDialog::OnOK();\n}\n\n\n\nvoid CDisplayTextSettingDlg::OnBnClickedRestoreDefaultButton()\n{\n    // TODO: 在此添加控件通知处理程序代码\n    int item_count = m_list_ctrl.GetItemCount();\n    for (int i{}; i < item_count; i++)\n    {\n        CommonDisplayItem display_item = GetDisplayItem(i);\n        CString default_text;\n        if (display_item.is_plugin)\n        {\n            default_text = display_item.plugin_item->GetItemLableText();\n        }\n        else\n        {\n            switch (display_item.item_type)\n            {\n            case TDI_UP:\n                if (m_main_window_text)\n                    default_text = CCommon::LoadText(IDS_UPLOAD_DISP, _T(\": \"));\n                else\n                    default_text = _T(\"↑: \");\n                break;\n            case TDI_DOWN:\n                if (m_main_window_text)\n                    default_text = CCommon::LoadText(IDS_DOWNLOAD_DISP, _T(\": \"));\n                else\n                    default_text = _T(\"↓: \");\n                break;\n            case TDI_TOTAL_SPEED:\n                default_text = _T(\"↑↓: \");\n                break;\n            case TDI_CPU:\n                default_text = _T(\"CPU: \");\n                break;\n            case TDI_CPU_FREQ:\n                default_text = CCommon::LoadText(IDS_CPU_FREQ, _T(\": \"));\n                break;\n            case TDI_MEMORY:\n                default_text = CCommon::LoadText(IDS_MEMORY_DISP, _T(\": \"));\n                break;\n            case TDI_GPU_USAGE:\n                default_text = CCommon::LoadText(IDS_GPU_DISP, _T(\": \"));\n                break;\n            case TDI_CPU_TEMP:\n                default_text = _T(\"CPU: \");\n                break;\n            case TDI_GPU_TEMP:\n                default_text = CCommon::LoadText(IDS_GPU_DISP, _T(\": \"));\n                break;\n            case TDI_HDD_TEMP:\n                default_text = CCommon::LoadText(IDS_HDD_DISP, _T(\": \"));\n                break;\n            case TDI_MAIN_BOARD_TEMP:\n                default_text = CCommon::LoadText(IDS_MAINBOARD_DISP, _T(\": \"));\n                break;\n            case TDI_HDD_USAGE:\n                default_text = CCommon::LoadText(IDS_HDD_DISP, _T(\": \"));\n                break;\n            default:\n                break;\n            }\n        }\n        m_list_ctrl.SetItemText(i, 1, default_text);\n    }\n}\n"
  },
  {
    "path": "TrafficMonitor/DisplayTextSettingDlg.h",
    "content": "﻿#pragma once\n#include \"BaseDialog.h\"\n#include \"CommonData.h\"\n#include \"ListCtrlEx.h\"\n\n// CDisplayTextSettingDlg 对话框\n\nclass CDisplayTextSettingDlg : public CBaseDialog\n{\n\tDECLARE_DYNAMIC(CDisplayTextSettingDlg)\n\npublic:\n\tCDisplayTextSettingDlg(DispStrings& display_texts, bool main_window_text = false, CWnd* pParent = nullptr);   // 标准构造函数\n\tvirtual ~CDisplayTextSettingDlg();\n\n// 对话框数据\n#ifdef AFX_DESIGN_TIME\n\tenum { IDD = IDD_DISPLAY_TEXT_SETTINGS_DIALOG };\n#endif\n\nprivate:\n    DispStrings& m_display_texts;\n\n    CListCtrlEx m_list_ctrl;\n    bool m_main_window_text{ false };       //如果为true，则为主窗口文本设置，否则为任务栏窗口设置\n\nprotected:\n    virtual CString GetDialogName() const override;\n    CommonDisplayItem GetDisplayItem(int row);\n\n\tvirtual void DoDataExchange(CDataExchange* pDX);    // DDX/DDV 支持\n\n\tDECLARE_MESSAGE_MAP()\n\npublic:\n    virtual BOOL OnInitDialog();\n    virtual void OnOK();\n    afx_msg void OnBnClickedRestoreDefaultButton();\n};\n"
  },
  {
    "path": "TrafficMonitor/DonateDlg.cpp",
    "content": "﻿// DonateDlg.cpp : 实现文件\n//\n\n#include \"stdafx.h\"\n#include \"TrafficMonitor.h\"\n#include \"DonateDlg.h\"\n#include \"afxdialogex.h\"\n#include \"DrawCommon.h\"\n\n\n// CDonateDlg 对话框\n\nIMPLEMENT_DYNAMIC(CDonateDlg, CBaseDialog)\n\nCDonateDlg::CDonateDlg(CWnd* pParent /*=NULL*/)\n    : CBaseDialog(IDD_DONATE_DIALOG, pParent)\n{\n\n}\n\nCDonateDlg::~CDonateDlg()\n{\n}\n\nvoid CDonateDlg::DoDataExchange(CDataExchange* pDX)\n{\n    CBaseDialog::DoDataExchange(pDX);\n    //DDX_Control(pDX, IDC_DONATE_PIC, m_donate_pic);\n}\n\n\nCString CDonateDlg::GetDialogName() const\n{\n    return _T(\"DonateDlg\");\n}\n\nBEGIN_MESSAGE_MAP(CDonateDlg, CBaseDialog)\n    ON_WM_PAINT()\nEND_MESSAGE_MAP()\n\n\n// CDonateDlg 消息处理程序\n\n\nBOOL CDonateDlg::OnInitDialog()\n{\n    CBaseDialog::OnInitDialog();\n\n    // TODO:  在此添加额外的初始化\n    SetWindowText(CCommon::LoadText(IDS_TITLE_DONATE));\n\n    //计算两个二维码图片的位置\n    CRect rect{};\n    CWnd* pWnd = nullptr;\n    pWnd = GetDlgItem(IDC_TEXT_STATIC);\n    if (pWnd != nullptr)\n    {\n        pWnd->GetWindowRect(rect);\n        ScreenToClient(rect);\n    }\n\n    CRect rc_pic_area;\n    GetClientRect(rc_pic_area);\n    rc_pic_area.top = rect.bottom + theApp.DPI(8);\n    rc_pic_area.left += theApp.DPI(10);\n    rc_pic_area.right -= theApp.DPI(10);\n    pWnd = GetDlgItem(IDOK);\n    if (pWnd != nullptr)\n    {\n        pWnd->GetWindowRect(rect);\n        ScreenToClient(rect);\n    }\n    rc_pic_area.bottom = rect.top - theApp.DPI(8);\n    m_pic1_rect = rc_pic_area;\n    m_pic1_rect.right = m_pic1_rect.left + (rc_pic_area.Width() / 2) - theApp.DPI(4);\n\n    m_pic2_rect = rc_pic_area;\n    m_pic2_rect.left = m_pic2_rect.right - (rc_pic_area.Width() / 2) + theApp.DPI(4);\n\n    return TRUE;  // return TRUE unless you set the focus to a control\n                  // 异常: OCX 属性页应返回 FALSE\n}\n\n\nvoid CDonateDlg::OnPaint()\n{\n    CPaintDC dc(this); // device context for painting\n                       // TODO: 在此处添加消息处理程序代码\n                       // 不为绘图消息调用 CBaseDialog::OnPaint()\n    CDrawCommon draw;\n    draw.Create(&dc, this);\n    draw.DrawBitmap(IDB_DONATE_BITMAP, m_pic1_rect.TopLeft(), m_pic1_rect.Size(), CDrawCommon::StretchMode::FIT);\n    draw.DrawBitmap(IDB_DONATE_WECHAT, m_pic2_rect.TopLeft(), m_pic2_rect.Size(), CDrawCommon::StretchMode::FIT);\n}\n"
  },
  {
    "path": "TrafficMonitor/DonateDlg.h",
    "content": "﻿#pragma once\n#include \"afxwin.h\"\n#include \"PictureStatic.h\"\n#include \"BaseDialog.h\"\n\n// CDonateDlg 对话框\n\nclass CDonateDlg : public CBaseDialog\n{\n    DECLARE_DYNAMIC(CDonateDlg)\n\npublic:\n    CDonateDlg(CWnd* pParent = NULL);   // 标准构造函数\n    virtual ~CDonateDlg();\n\n    // 对话框数据\n#ifdef AFX_DESIGN_TIME\n    enum { IDD = IDD_DONATE_DIALOG };\n#endif\n\nprotected:\n    //CPictureStatic m_donate_pic;\n    CRect m_pic1_rect;\n    CRect m_pic2_rect;\n\n    virtual void DoDataExchange(CDataExchange* pDX);    // DDX/DDV 支持\n    virtual CString GetDialogName() const override;\n\n    DECLARE_MESSAGE_MAP()\n\npublic:\n    virtual BOOL OnInitDialog();\n    afx_msg void OnPaint();\n};\n"
  },
  {
    "path": "TrafficMonitor/DrawCommon.cpp",
    "content": "﻿#include \"stdafx.h\"\n#include \"DrawCommon.h\"\n\n\nCDrawCommon::CDrawCommon()\n{\n}\n\nCDrawCommon::~CDrawCommon()\n{\n}\n\nvoid CDrawCommon::Create(CDC* pDC, CWnd* pMainWnd)\n{\n    m_pDC = pDC;\n    m_pMainWnd = pMainWnd;\n    if (pMainWnd != nullptr)\n        m_pfont = m_pMainWnd->GetFont();\n}\n\nvoid CDrawCommon::SetFont(CFont* pfont)\n{\n    m_pfont = pfont;\n    m_pDC->SelectObject(m_pfont);\n}\n\nvoid CDrawCommon::SetDC(CDC* pDC)\n{\n    m_pDC = pDC;\n}\n\nvoid CDrawCommon::DrawWindowText(CRect rect, LPCTSTR lpszString, COLORREF color, Alignment align, bool draw_back_ground, bool multi_line)\n{\n    m_pDC->SetTextColor(color);\n    if (!draw_back_ground)\n        m_pDC->SetBkMode(TRANSPARENT);\n    m_pDC->SelectObject(m_pfont);\n    CSize text_size = m_pDC->GetTextExtent(lpszString);\n\n    UINT format;        //CDC::DrawText()函数的文本格式\n    if (multi_line)\n        format = DT_EDITCONTROL | DT_WORDBREAK | DT_NOPREFIX;\n    else\n        format = DT_VCENTER | DT_SINGLELINE | DT_NOPREFIX;\n\n    if (text_size.cx > rect.Width())        //如果文本宽度超过了矩形区域的宽度，设置了居中时左对齐\n    {\n        if (align == Alignment::RIGHT)\n            format |= DT_RIGHT;\n    }\n    else\n    {\n        switch (align)\n        {\n        case Alignment::RIGHT: format |= DT_RIGHT; break;\n        case Alignment::CENTER: format |= DT_CENTER; break;\n        }\n    }\n    if (draw_back_ground)\n        m_pDC->FillSolidRect(rect, m_back_color);\n    m_pDC->DrawText(lpszString, rect, format);\n}\n\n\nvoid CDrawCommon::SetDrawRect(CRect rect)\n{\n    CRgn rgn;\n    rgn.CreateRectRgnIndirect(rect);\n    m_pDC->SelectClipRgn(&rgn);\n}\n\nvoid CDrawCommon::SetDrawRect(CDC* pDC, CRect rect)\n{\n    CRgn rgn;\n    rgn.CreateRectRgnIndirect(rect);\n    pDC->SelectClipRgn(&rgn);\n}\n\nvoid CDrawCommon::DrawBitmap(CBitmap& bitmap, CPoint start_point, CSize size, StretchMode stretch_mode)\n{\n    CDC memDC;\n\n    //获取图像实际大小\n    BITMAP bm;\n    GetObject(bitmap, sizeof(BITMAP), &bm);\n\n    memDC.CreateCompatibleDC(m_pDC);\n    memDC.SelectObject(&bitmap);\n    // 以下两行避免图片失真\n    m_pDC->SetStretchBltMode(HALFTONE);\n    m_pDC->SetBrushOrg(0, 0);\n    CSize draw_size;\n    if (size.cx == 0 || size.cy == 0)       //如果指定的size为0，则使用位图的实际大小绘制\n    {\n        draw_size = CSize(bm.bmWidth, bm.bmHeight);\n    }\n    else\n    {\n        draw_size = size;\n        if (stretch_mode == StretchMode::FILL)\n        {\n            SetDrawRect(CRect(start_point, draw_size));\n            float w_h_radio, w_h_radio_draw;        //图像的宽高比、绘制大小的宽高比\n            w_h_radio = static_cast<float>(bm.bmWidth) / bm.bmHeight;\n            w_h_radio_draw = static_cast<float>(size.cx) / size.cy;\n            if (w_h_radio > w_h_radio_draw)     //如果图像的宽高比大于绘制区域的宽高比，则需要裁剪两边的图像\n            {\n                int image_width;        //按比例缩放后的宽度\n                image_width = bm.bmWidth * draw_size.cy / bm.bmHeight;\n                start_point.x -= ((image_width - draw_size.cx) / 2);\n                draw_size.cx = image_width;\n            }\n            else\n            {\n                int image_height;       //按比例缩放后的高度\n                image_height = bm.bmHeight * draw_size.cx / bm.bmWidth;\n                start_point.y -= ((image_height - draw_size.cy) / 2);\n                draw_size.cy = image_height;\n            }\n        }\n        else if (stretch_mode == StretchMode::FIT)\n        {\n            draw_size = CSize(bm.bmWidth, bm.bmHeight);\n            float w_h_radio, w_h_radio_draw;        //图像的宽高比、绘制大小的宽高比\n            w_h_radio = static_cast<float>(bm.bmWidth) / bm.bmHeight;\n            w_h_radio_draw = static_cast<float>(size.cx) / size.cy;\n            if (w_h_radio > w_h_radio_draw)     //如果图像的宽高比大于绘制区域的宽高比\n            {\n                draw_size.cy = draw_size.cy * size.cx / draw_size.cx;\n                draw_size.cx = size.cx;\n                start_point.y += ((size.cy - draw_size.cy) / 2);\n            }\n            else\n            {\n                draw_size.cx = draw_size.cx * size.cy / draw_size.cy;\n                draw_size.cy = size.cy;\n                start_point.x += ((size.cx - draw_size.cx) / 2);\n            }\n        }\n    }\n\n    m_pDC->StretchBlt(start_point.x, start_point.y, draw_size.cx, draw_size.cy, &memDC, 0, 0, bm.bmWidth, bm.bmHeight, SRCCOPY);\n    memDC.DeleteDC();\n}\n\nvoid CDrawCommon::DrawBitmap(UINT bitmap_id, CPoint start_point, CSize size, StretchMode stretch_mode)\n{\n    CBitmap bitmap;\n    bitmap.LoadBitmap(bitmap_id);\n    DrawBitmap(bitmap, start_point, size, stretch_mode);\n}\n\nvoid CDrawCommon::DrawBitmap(HBITMAP hbitmap, CPoint start_point, CSize size, StretchMode stretch_mode)\n{\n    CBitmap bitmap;\n    if (!bitmap.Attach(hbitmap))\n        return;\n    DrawBitmap(bitmap, start_point, size, stretch_mode);\n    bitmap.Detach();\n}\n\nvoid CDrawCommon::DrawIcon(HICON hIcon, CPoint start_point, CSize size)\n{\n    if (m_pDC->GetSafeHdc() == NULL)\n        return;\n    if (size.cx == 0 || size.cy == 0)\n        ::DrawIconEx(m_pDC->GetSafeHdc(), start_point.x, start_point.y, hIcon, 0, 0, 0, NULL, DI_NORMAL | DI_DEFAULTSIZE);\n    else\n        ::DrawIconEx(m_pDC->GetSafeHdc(), start_point.x, start_point.y, hIcon, size.cx, size.cy, 0, NULL, DI_NORMAL);\n}\n\nvoid CDrawCommon::BitmapStretch(CImage* pImage, CImage* ResultImage, CSize size)\n{\n    if (pImage->IsDIBSection())\n    {\n        // 取得 pImage 的 DC\n        CDC* pImageDC1 = CDC::FromHandle(pImage->GetDC()); // Image 因為有自己的 DC, 所以必須使用 FromHandle 取得對應的 DC\n\n        CBitmap* bitmap1 = pImageDC1->GetCurrentBitmap();\n        BITMAP bmpInfo;\n        bitmap1->GetBitmap(&bmpInfo);\n\n        // 建立新的 CImage\n        ResultImage->Create(size.cx, size.cy, bmpInfo.bmBitsPixel);\n        CDC* ResultImageDC = CDC::FromHandle(ResultImage->GetDC());\n\n        // 當 Destination 比較小的時候, 會根據 Destination DC 上的 Stretch Blt mode 決定是否要保留被刪除點的資訊\n        ResultImageDC->SetStretchBltMode(HALFTONE); // 使用最高品質的方式\n        ::SetBrushOrgEx(ResultImageDC->m_hDC, 0, 0, NULL); // 調整 Brush 的起點\n\n        // 把 pImage 畫到 ResultImage 上面\n        StretchBlt(*ResultImageDC, 0, 0, size.cx, size.cy, *pImageDC1, 0, 0, pImage->GetWidth(), pImage->GetHeight(), SRCCOPY);\n        // pImage->Draw(*ResultImageDC,0,0,StretchWidth,StretchHeight,0,0,pImage->GetWidth(),pImage->GetHeight());\n\n        pImage->ReleaseDC();\n        ResultImage->ReleaseDC();\n    }\n}\n\nvoid CDrawCommon::FillRect(CRect rect, COLORREF color)\n{\n    m_pDC->FillSolidRect(rect, color);\n}\n\nvoid CDrawCommon::FillRectWithBackColor(CRect rect)\n{\n    m_pDC->FillSolidRect(rect, m_back_color);\n}\n\nvoid CDrawCommon::DrawRectOutLine(CRect rect, COLORREF color, int width, bool dot_line)\n{\n    CPen aPen, * pOldPen;\n    aPen.CreatePen((dot_line ? PS_DOT : PS_SOLID), width, color);\n    pOldPen = m_pDC->SelectObject(&aPen);\n    CBrush* pOldBrush{ dynamic_cast<CBrush*>(m_pDC->SelectStockObject(NULL_BRUSH)) };\n\n    rect.DeflateRect(width / 2, width / 2);\n    m_pDC->Rectangle(rect);\n    m_pDC->SelectObject(pOldPen);\n    m_pDC->SelectObject(pOldBrush);       // Restore the old brush\n    aPen.DeleteObject();\n}\n\nvoid CDrawCommon::GetRegionFromImage(CRgn& rgn, CBitmap& cBitmap, int threshold)\n{\n    CDC memDC;\n\n    memDC.CreateCompatibleDC(NULL);\n    CBitmap* pOldMemBmp = NULL;\n    pOldMemBmp = memDC.SelectObject(&cBitmap);\n\n    //创建总的窗体区域，初始region为0\n    rgn.CreateRectRgn(0, 0, 0, 0);\n\n    BITMAP bit;\n    cBitmap.GetBitmap(&bit);//取得位图参数，这里要用到位图的长和宽\n    int y;\n    for (y = 0; y < bit.bmHeight; y++)\n    {\n        CRgn rgnTemp; //保存临时region\n        int iX = 0;\n        do\n        {\n            //跳过透明色找到下一个非透明色的点.\n            while (iX < bit.bmWidth && GetColorBritness(memDC.GetPixel(iX, y)) <= threshold)\n                iX++;\n            int iLeftX = iX; //记住这个起始点\n\n                             //寻找下个透明色的点\n            while (iX < bit.bmWidth && GetColorBritness(memDC.GetPixel(iX, y)) > threshold)\n                ++iX;\n\n            //创建一个包含起点与重点间高为1像素的临时“region”\n            rgnTemp.CreateRectRgn(iLeftX, y, iX, y + 1);\n            rgn.CombineRgn(&rgn, &rgnTemp, RGN_OR);\n\n            //删除临时\"region\",否则下次创建时和出错\n            rgnTemp.DeleteObject();\n        } while (iX < bit.bmWidth);\n    }\n    memDC.DeleteDC();\n}\n\nint CDrawCommon::GetColorBritness(COLORREF color)\n{\n    return (GetRValue(color) + GetGValue(color) + GetBValue(color)) / 3;\n}\n\nvoid CDrawCommon::DrawLine(CPoint start_point, int height, COLORREF color)\n{\n    CPen aPen, * pOldPen;\n    aPen.CreatePen(PS_SOLID, 1, color);\n    pOldPen = m_pDC->SelectObject(&aPen);\n    CBrush* pOldBrush{ dynamic_cast<CBrush*>(m_pDC->SelectStockObject(NULL_BRUSH)) };\n\n    m_pDC->MoveTo(start_point); //移动到起始点，默认是从下向上画\n    m_pDC->LineTo(CPoint(start_point.x, start_point.y - height));\n    m_pDC->SelectObject(pOldPen);\n    m_pDC->SelectObject(pOldBrush);       // Restore the old brush\n    aPen.DeleteObject();\n}\n"
  },
  {
    "path": "TrafficMonitor/DrawCommon.h",
    "content": "﻿//封装的绘图类\n#pragma once\n#include \"CommonData.h\"\nclass CDrawCommon\n{\npublic:\n\n    //拉伸模式\n    enum class StretchMode\n    {\n        STRETCH,\t\t//拉伸，会改变比例\n        FILL,\t\t\t//填充，不改变比例，会裁剪长边\n        FIT\t\t\t//适应，不会改变比例，不裁剪\n    };\n\n    CDrawCommon();\n    ~CDrawCommon();\n\n    void Create(CDC* pDC, CWnd* pMainWnd);\n    void SetFont(CFont* pfont);\t\t//设置绘制文本的字体\n    void SetDC(CDC* pDC);\t\t//设置绘图的DC\n    CDC* GetDC() { return m_pDC; }\n    void SetBackColor(COLORREF back_color) { m_back_color = back_color; }\n\n    void DrawWindowText(CRect rect, LPCTSTR lpszString, COLORREF color, Alignment align = Alignment::LEFT, bool draw_back_ground = false, bool multi_line = false);\t//在指定的矩形区域内绘制文本\n\n    void SetDrawRect(CRect rect);\t\t//设置绘图剪辑区域\n    static void SetDrawRect(CDC* pDC, CRect rect);\n\n    //绘制一个位图\n    //（注意：当stretch_mode设置为StretchMode::FILL（填充）时，会设置绘图剪辑区域，如果之后需要绘制其他图形，\n    //需要重新设置绘图剪辑区域，否则图片外的区域会无法绘制）\n    void DrawBitmap(CBitmap& bitmap, CPoint start_point, CSize size, StretchMode stretch_mode = StretchMode::STRETCH);\n    void DrawBitmap(UINT bitmap_id, CPoint start_point, CSize size, StretchMode stretch_mode = StretchMode::STRETCH);\n    void DrawBitmap(HBITMAP hbitmap, CPoint start_point, CSize size, StretchMode stretch_mode = StretchMode::STRETCH);\n\n    void DrawIcon(HICON hIcon, CPoint start_point, CSize size);\n\n    //将图片拉伸到指定尺寸(https://blog.csdn.net/sichuanpb/article/details/22986877)\n    static void BitmapStretch(CImage* pImage, CImage* ResultImage, CSize size);\n\n    void FillRect(CRect rect, COLORREF color);\t\t//用纯色填充矩形\n    void FillRectWithBackColor(CRect rect);\t\t\t//使用背景色填充矩形\n    void DrawRectOutLine(CRect rect, COLORREF color, int width = 1, bool dot_line = false);\t//绘制矩形边框。如果dot_line为true，则为虚线\n\n    //从图像创建区域，如果像素点的亮度小于threshold（取值为0~255，0为黑色，255为白色），则该像素点在区域外\n    //https://blog.csdn.net/tajon1226/article/details/6589180\n    static void GetRegionFromImage(CRgn& rgn, CBitmap& cBitmap, int threshold);\n\n    void DrawLine(CPoint start_point, int height, COLORREF color);  //使用当前画笔画线\n\nprivate:\n    CDC* m_pDC{};\t\t//用于绘图的CDC类的指针\n    CWnd* m_pMainWnd{};\t//绘图窗口的句柄\n    CFont* m_pfont{};\n    COLORREF m_back_color{};\n\n    static int GetColorBritness(COLORREF color);\n};\n\n\n//用于双缓冲绘图的类\nclass CDrawDoubleBuffer\n{\npublic:\n    CDrawDoubleBuffer(CDC* pDC, CRect rect)\n        : m_pDC(pDC), m_rect(rect)\n    {\n        if (m_pDC != nullptr)\n        {\n            m_memDC.CreateCompatibleDC(NULL);\n            m_memBitmap.CreateCompatibleBitmap(pDC, rect.Width(), rect.Height());\n            m_pOldBit = m_memDC.SelectObject(&m_memBitmap);\n        }\n    }\n\n    ~CDrawDoubleBuffer()\n    {\n        if (m_pDC != nullptr)\n        {\n            m_pDC->BitBlt(m_rect.left, m_rect.top, m_rect.Width(), m_rect.Height(), &m_memDC, 0, 0, SRCCOPY);\n            m_memDC.SelectObject(m_pOldBit);\n            m_memBitmap.DeleteObject();\n            m_memDC.DeleteDC();\n        }\n    }\n\n    CDC* GetMemDC()\n    {\n        return &m_memDC;\n    }\n\nprivate:\n    CDC* m_pDC;\n    CDC m_memDC;\n    CBitmap m_memBitmap;\n    CBitmap* m_pOldBit;\n    CRect m_rect;\n};\n"
  },
  {
    "path": "TrafficMonitor/DrawCommonEx.cpp",
    "content": "#include \"stdafx.h\"\n#include \"DrawCommonEx.h\"\n\n\nCDrawCommonEx::CDrawCommonEx(CDC* pDC)\n{\n\tm_pDC = pDC;\n\thThm = OpenThemeData(::GetDesktopWindow(), L\"Window\");\n}\n\n\nCDrawCommonEx::~CDrawCommonEx()\n{\n\tCloseThemeData(hThm);\n}\n\nvoid CDrawCommonEx::DrawWindowText(CRect rect, LPCTSTR str, COLORREF color)\n{\n\tm_pDC->SelectObject(m_pFont);\n\tm_pDC->SetTextColor(color);\n\n\tDTTOPTS dttopts{};\n\tdttopts.dwSize = sizeof(DTTOPTS);\n\n\tdttopts.dwFlags = DTT_GLOWSIZE | DTT_COMPOSITED; //ѡ \n\tdttopts.iGlowSize = 0; //ķΧС \n\tHRESULT hr = DrawThemeTextEx(hThm, m_pDC->GetSafeHdc(), TEXT_LABEL, 0, str, -1, DT_LEFT | DT_SINGLELINE, rect, &dttopts);\n}\n\nvoid CDrawCommonEx::SetFont(CFont * pFont)\n{\n\tm_pFont = pFont;\n}\n"
  },
  {
    "path": "TrafficMonitor/DrawCommonEx.h",
    "content": "#pragma once\nclass CDrawCommonEx\n{\npublic:\n\tCDrawCommonEx(CDC* pDC);\n\t~CDrawCommonEx();\n\n\tvoid DrawWindowText(CRect rect, LPCTSTR str, COLORREF color);\n\tvoid SetFont(CFont* pFont);\n\nprivate:\n\tHTHEME hThm;\n\tCFont* m_pFont{};\n\tCDC* m_pDC{};\n};\n\n"
  },
  {
    "path": "TrafficMonitor/FilePathHelper.cpp",
    "content": "﻿#include \"stdafx.h\"\n#include \"FilePathHelper.h\"\n#include \"Common.h\"\n\n\nCFilePathHelper::CFilePathHelper(const wstring & file_path)\n\t: m_file_path{ file_path }\n{\n}\n\nCFilePathHelper::~CFilePathHelper()\n{\n}\n\nwstring CFilePathHelper::GetFileExtension(bool upper, bool width_dot) const\n{\n\tsize_t index;\n\tindex = m_file_path.rfind('.');\n\tif (index == wstring::npos || index == m_file_path.size() - 1)\n\t\treturn wstring();\n\twstring file_extension{ m_file_path.substr(width_dot ? index : index + 1) };\n\tCCommon::StringTransform(file_extension, upper);\n\treturn file_extension;\n}\n\nwstring CFilePathHelper::GetFileName() const\n{\n\tsize_t index;\n\tindex = m_file_path.rfind('\\\\');\n\tif (index == wstring::npos)\n\t\tindex = m_file_path.rfind('/');\n\treturn m_file_path.substr(index + 1);\n}\n\nwstring CFilePathHelper::GetFileNameWithoutExtension() const\n{\n\tsize_t index, index1;\n\tindex = m_file_path.rfind('.');\n\tindex1 = m_file_path.rfind('\\\\');\n\tif (index1 == wstring::npos)\n\t\tindex1 = m_file_path.rfind('/');\n\treturn m_file_path.substr(index1 + 1, (index - index1 - 1));\n}\n\nwstring CFilePathHelper::GetFolderName() const\n{\n\tint index, index1;\n\tindex = m_file_path.find_last_of(L\"\\\\/\");\n\tif (index == wstring::npos || index == 0)\n\t\treturn wstring();\n\n\tindex1 = m_file_path.find_last_of(L\"\\\\/\", index - 1);\n\tif (index1 == wstring::npos || index1 == 0)\n\t\treturn wstring();\n\n\treturn m_file_path.substr(index1 + 1, (index - index1 - 1));\n}\n\nwstring CFilePathHelper::GetDir() const\n{\n\tif (!m_file_path.empty() && (m_file_path.back() == L'\\\\' || m_file_path.back() == L'/'))\n\t\treturn m_file_path;\n\tsize_t index;\n\tindex = m_file_path.rfind('\\\\');\n\tif (index == wstring::npos)\n\t\tindex = m_file_path.rfind('/');\n\treturn m_file_path.substr(0, index + 1);\n}\n\nwstring CFilePathHelper::GetParentDir() const\n{\n\twstring dir{ GetDir() };\n\tsize_t index;\n\tif (!dir.empty() && (dir.back() == L'\\\\' || dir.back() == L'/'))\n\t\tdir.pop_back();\n\tindex = dir.rfind('\\\\');\n\tif (index == wstring::npos)\n\t\tindex = dir.rfind('/');\n\treturn m_file_path.substr(0, index + 1);\n}\n\nconst wstring& CFilePathHelper::ReplaceFileExtension(const wchar_t * new_extension)\n{\n\tsize_t index, index1;\n\tindex = m_file_path.rfind('.');\n\tindex1 = m_file_path.rfind('\\\\');\n\tif (index == wstring::npos || (index1 != wstring::npos && index < index1))\t\t//如果没有找到“.”，或者“.”在反斜杠的左边，则在末尾添加一个“.”\n\t{\n\t\tm_file_path.push_back(L'.');\n\t}\n\telse if (index != m_file_path.size() - 1)\t//如果“.”不在最后的位置，则删除“.”后面的字符串\n\t{\n\t\tm_file_path.erase(index + 1);\n\t}\n    if (new_extension == nullptr || *new_extension == L'\\0')\n    {\n        if (!m_file_path.empty() && m_file_path.back() == L'.')\n            m_file_path.pop_back();\n    }\n    else\n    {\n        m_file_path.append(new_extension);\t\t//在末尾添加扩展名\n    }\n\treturn m_file_path;\n}\n\nwstring CFilePathHelper::GetFilePathWithoutExtension() const\n{\n    size_t index;\n    index = m_file_path.rfind('.');\n    return m_file_path.substr(0, index);\n}\n"
  },
  {
    "path": "TrafficMonitor/FilePathHelper.h",
    "content": "﻿#pragma once\nclass CFilePathHelper\n{\npublic:\n\tCFilePathHelper(const wstring& file_path);\n\tCFilePathHelper(){}\n\t~CFilePathHelper();\n\n\tvoid SetFilePath(const wstring& file_path) { m_file_path = file_path; }\n\n\twstring GetFileExtension(bool upper = false, bool width_dot = false) const;\t\t//获取文件的扩展名(upper:是否大写; width_dot:是否包含“.”)\n\twstring GetFileName() const;\t\t\t\t\t\t\t//获取文件名\n\twstring GetFileNameWithoutExtension() const;\t\t\t//获取文件名（不含扩展名）\n\twstring GetFolderName() const;\t\t\t\t\t\t\t//获取文件夹名\n\twstring GetDir() const;\t\t\t\t\t\t\t\t\t//获取目录\n\twstring GetParentDir() const;\t\t\t\t\t\t\t//获取上级目录\n\twstring GetFilePath() const { return m_file_path; }\t\t//获取完整路径\n\tconst wstring& ReplaceFileExtension(const wchar_t* new_extension);\t\t//替换文件的扩展名，返回文件完整路径\n    wstring GetFilePathWithoutExtension() const;            //获取文件路径（不含扩展名）\nprotected:\n\twstring m_file_path;\n};\n\n"
  },
  {
    "path": "TrafficMonitor/GeneralSettingsDlg.cpp",
    "content": "﻿// GeneralSettingsDlg.cpp : implementation file\n//\n\n#include \"stdafx.h\"\n#include \"TrafficMonitor.h\"\n#include \"GeneralSettingsDlg.h\"\n#include \"PluginManagerDlg.h\"\n#include \"SelectConnectionsDlg.h\"\n\n\n// CGeneralSettingsDlg dialog\n\nstatic const int MONITOR_SPAN_STEP = 100;\n\nIMPLEMENT_DYNAMIC(CGeneralSettingsDlg, CTabDlg)\n\nCGeneralSettingsDlg::CGeneralSettingsDlg(CWnd* pParent /*=NULL*/)\n    : CTabDlg(IDD_GENERAL_SETTINGS_DIALOG, pParent)\n{\n\n}\n\nCGeneralSettingsDlg::~CGeneralSettingsDlg()\n{\n}\n\nvoid CGeneralSettingsDlg::CheckTaskbarDisplayItem()\n{\n    //如果选项设置中关闭了某个硬件监控，则不显示对应的温度监控相关项目\n    int taskbar_displat_item_ori = theApp.m_taskbar_data.m_tbar_display_item;\n    if (!theApp.m_general_data.IsHardwareEnable(HI_CPU))\n    {\n        theApp.m_taskbar_data.m_tbar_display_item &= ~TDI_CPU_TEMP;\n        theApp.m_taskbar_data.m_tbar_display_item &= ~TDI_CPU_FREQ;\n    }\n    if (!theApp.m_general_data.IsHardwareEnable(HI_GPU))\n    {\n        theApp.m_taskbar_data.m_tbar_display_item &= ~TDI_GPU_USAGE;\n        theApp.m_taskbar_data.m_tbar_display_item &= ~TDI_GPU_TEMP;\n    }\n    if (!theApp.m_general_data.IsHardwareEnable(HI_HDD))\n    {\n        theApp.m_taskbar_data.m_tbar_display_item &= ~TDI_HDD_TEMP;\n        theApp.m_taskbar_data.m_tbar_display_item &= ~TDI_HDD_USAGE;\n    }\n    if (!theApp.m_general_data.IsHardwareEnable(HI_MBD))\n        theApp.m_taskbar_data.m_tbar_display_item &= ~TDI_MAIN_BOARD_TEMP;\n}\n\nvoid CGeneralSettingsDlg::SetControlMouseWheelEnable(bool enable)\n{\n    m_traffic_tip_combo.SetMouseWheelEnable(enable);\n    m_language_combo.SetMouseWheelEnable(enable);\n    m_traffic_tip_edit.SetMouseWheelEnable(enable);\n    m_memory_tip_edit.SetMouseWheelEnable(enable);\n    m_monitor_span_edit.SetMouseWheelEnable(enable);\n    m_cpu_temp_tip_edit.SetMouseWheelEnable(enable);\n    m_gpu_temp_tip_edit.SetMouseWheelEnable(enable);\n    m_hdd_temp_tip_edit.SetMouseWheelEnable(enable);\n    m_mbd_temp_tip_edit.SetMouseWheelEnable(enable);\n    m_hard_disk_combo.SetMouseWheelEnable(enable);\n    m_select_cpu_combo.SetMouseWheelEnable(enable);\n}\n\nbool CGeneralSettingsDlg::ShowHardwareMonitorWarning()\n{\n    if (SHMessageBoxCheck(m_hWnd, CCommon::LoadText(IDS_HARDWARE_MONITOR_WARNING), APP_NAME, MB_OKCANCEL | MB_ICONWARNING, IDOK, _T(\"{B8A281A7-76DF-4F0F-BF6A-1A394EF8BAD5}\")) == IDOK)\n    {\n        //if (SHMessageBoxCheck(m_hWnd, CCommon::LoadText(IDS_HARDWARE_MONITOR_WARNING2), APP_NAME, MB_OKCANCEL | MB_ICONWARNING, IDOK, _T(\"{2777F260-6175-41E4-AF59-4085B3F58E32}\")) == IDOK)\n        //{\n        return true;\n        //}\n    }\n    return false;\n}\n\nvoid CGeneralSettingsDlg::AddOrUpdateAutoRunTooltip(bool add)\n{\n    CString str_tool_tip;\n#ifdef WITHOUT_TEMPERATURE\n    str_tool_tip = CCommon::LoadText(IDS_AUTO_RUN_METHOD_REGESTRY);\n#else\n    str_tool_tip = CCommon::LoadText(IDS_AUTO_RUN_METHOD_TASK_SCHEDULE);\n#endif\n    if (!m_auto_run_path.empty())\n    {\n        str_tool_tip += _T(\"\\r\\n\");\n        str_tool_tip += CCommon::LoadText(IDS_PATH, _T(\": \"));\n        str_tool_tip += m_auto_run_path.c_str();\n    }\n    if (add)\n        m_toolTip.AddTool(GetDlgItem(IDC_AUTO_RUN_CHECK), str_tool_tip);\n    else\n        m_toolTip.UpdateTipText(str_tool_tip, GetDlgItem(IDC_AUTO_RUN_CHECK));\n\n}\n\nbool CGeneralSettingsDlg::IsMonitorTimeSpanModified() const\n{\n    return m_data.monitor_time_span != m_monitor_time_span_ori;\n}\n\nvoid CGeneralSettingsDlg::DoDataExchange(CDataExchange* pDX)\n{\n    CTabDlg::DoDataExchange(pDX);\n    DDX_Control(pDX, IDC_TODAY_TRAFFIC_TIP_EDIT, m_traffic_tip_edit);\n    DDX_Control(pDX, IDC_TODAY_TRAFFIC_TIP_COMBO, m_traffic_tip_combo);\n    DDX_Control(pDX, IDC_MEMORY_USAGE_TIP_EDIT, m_memory_tip_edit);\n    DDX_Control(pDX, IDC_LANGUAGE_COMBO, m_language_combo);\n    DDX_Control(pDX, IDC_MONITOR_SPAN_EDIT, m_monitor_span_edit);\n    DDX_Control(pDX, IDC_CPU_TEMP_TIP_EDIT, m_cpu_temp_tip_edit);\n    DDX_Control(pDX, IDC_GPU_TEMP_TIP_EDIT, m_gpu_temp_tip_edit);\n    DDX_Control(pDX, IDC_HDD_TIP_EDIT, m_hdd_temp_tip_edit);\n    DDX_Control(pDX, IDC_MBD_TEMP_TIP_EDIT, m_mbd_temp_tip_edit);\n    DDX_Control(pDX, IDC_SELECT_HARD_DISK_COMBO, m_hard_disk_combo);\n    DDX_Control(pDX, IDC_SELECT_CPU_COMBO, m_select_cpu_combo);\n    DDX_Control(pDX, IDC_PLUGIN_MANAGE_BUTTON, m_plugin_manager_btn);\n    DDX_Control(pDX, IDC_SELECT_CONNECTIONS_BUTTON, m_select_connection_btn);\n}\n\nvoid CGeneralSettingsDlg::SetControlEnable()\n{\n    m_traffic_tip_edit.EnableWindow(m_data.traffic_tip_enable);\n    m_traffic_tip_combo.EnableWindow(m_data.traffic_tip_enable);\n    m_memory_tip_edit.EnableWindow(m_data.memory_usage_tip.enable);\n    m_cpu_temp_tip_edit.EnableWindow(m_data.cpu_temp_tip.enable);\n    m_gpu_temp_tip_edit.EnableWindow(m_data.gpu_temp_tip.enable);\n    m_hdd_temp_tip_edit.EnableWindow(m_data.hdd_temp_tip.enable);\n    m_mbd_temp_tip_edit.EnableWindow(m_data.mainboard_temp_tip.enable);\n\n    m_hard_disk_combo.EnableWindow(m_data.IsHardwareEnable(HI_HDD));\n    m_select_cpu_combo.EnableWindow(m_data.IsHardwareEnable(HI_CPU));\n}\n\n\nBEGIN_MESSAGE_MAP(CGeneralSettingsDlg, CTabDlg)\n    ON_BN_CLICKED(IDC_CHECK_NOW_BUTTON, &CGeneralSettingsDlg::OnBnClickedCheckNowButton)\n    ON_BN_CLICKED(IDC_CHECK_UPDATE_CHECK, &CGeneralSettingsDlg::OnBnClickedCheckUpdateCheck)\n    ON_BN_CLICKED(IDC_AUTO_RUN_CHECK, &CGeneralSettingsDlg::OnBnClickedAutoRunCheck)\n    ON_BN_CLICKED(IDC_ALLOW_SKIN_FONT_CHECK, &CGeneralSettingsDlg::OnBnClickedAllowSkinFontCheck)\n    ON_BN_CLICKED(IDC_ALLOW_SKIN_DISP_STR_CHECK, &CGeneralSettingsDlg::OnBnClickedAllowSkinDispStrCheck)\n    ON_BN_CLICKED(IDC_TODAY_TRAFFIC_TIP_CHECK, &CGeneralSettingsDlg::OnBnClickedTodayTrafficTipCheck)\n    ON_BN_CLICKED(IDC_MEMORY_USAGE_TIP_CHECK, &CGeneralSettingsDlg::OnBnClickedMemoryUsageTipCheck)\n    ON_BN_CLICKED(IDC_OPEN_CONFIG_PATH_BUTTON, &CGeneralSettingsDlg::OnBnClickedOpenConfigPathButton)\n    ON_BN_CLICKED(IDC_SHOW_ALL_CONNECTION_CHECK, &CGeneralSettingsDlg::OnBnClickedShowAllConnectionCheck)\n    ON_BN_CLICKED(IDC_USE_CPU_TIME_RADIO, &CGeneralSettingsDlg::OnBnClickedUseCpuTimeRadio)\n    ON_BN_CLICKED(IDC_USE_PDH_RADIO, &CGeneralSettingsDlg::OnBnClickedUsePdhRadio)\n    ON_NOTIFY(UDN_DELTAPOS, SPIN_ID, &CGeneralSettingsDlg::OnDeltaposSpin)\n    ON_EN_KILLFOCUS(IDC_MONITOR_SPAN_EDIT, &CGeneralSettingsDlg::OnEnKillfocusMonitorSpanEdit)\n    ON_BN_CLICKED(IDC_CPU_TEMP_TIP_CHECK, &CGeneralSettingsDlg::OnBnClickedCpuTempTipCheck)\n    ON_BN_CLICKED(IDC_GPU_TEMP_TIP_CHECK, &CGeneralSettingsDlg::OnBnClickedGpuTempTipCheck)\n    ON_BN_CLICKED(IDC_HDD_TEMP_TIP_CHECK, &CGeneralSettingsDlg::OnBnClickedHddTempTipCheck)\n    ON_BN_CLICKED(IDC_MBD_TEMP_TIP_CHECK, &CGeneralSettingsDlg::OnBnClickedMbdTempTipCheck)\n    ON_BN_CLICKED(IDC_GITHUB_RADIO, &CGeneralSettingsDlg::OnBnClickedGithubRadio)\n    ON_BN_CLICKED(IDC_GITEE_RADIO, &CGeneralSettingsDlg::OnBnClickedGiteeRadio)\n    ON_BN_CLICKED(IDC_RESTORE_DEFAULT_TIME_SPAN_BUTTON, &CGeneralSettingsDlg::OnBnClickedRestoreDefaultTimeSpanButton)\n    ON_CBN_SELCHANGE(IDC_SELECT_HARD_DISK_COMBO, &CGeneralSettingsDlg::OnCbnSelchangeSelectHardDiskCombo)\n    ON_BN_CLICKED(IDC_CPU_CHECK, &CGeneralSettingsDlg::OnBnClickedCpuCheck)\n    ON_BN_CLICKED(IDC_GPU_CHECK, &CGeneralSettingsDlg::OnBnClickedGpuCheck)\n    ON_BN_CLICKED(IDC_HDD_CHECK, &CGeneralSettingsDlg::OnBnClickedHddCheck)\n    ON_BN_CLICKED(IDC_MBD_CHECK, &CGeneralSettingsDlg::OnBnClickedMbdCheck)\n    ON_CBN_SELCHANGE(IDC_SELECT_CPU_COMBO, &CGeneralSettingsDlg::OnCbnSelchangeSelectCpuCombo)\n    ON_BN_CLICKED(IDC_PLUGIN_MANAGE_BUTTON, &CGeneralSettingsDlg::OnBnClickedPluginManageButton)\n    ON_BN_CLICKED(IDC_SHOW_NOTIFY_ICON_CHECK, &CGeneralSettingsDlg::OnBnClickedShowNotifyIconCheck)\n    ON_BN_CLICKED(IDC_SELECT_CONNECTIONS_BUTTON, &CGeneralSettingsDlg::OnBnClickedSelectConnectionsButton)\n    ON_BN_CLICKED(IDC_RESET_AUTO_RUN_BUTTON, &CGeneralSettingsDlg::OnBnClickedResetAutoRunButton)\nEND_MESSAGE_MAP()\n\n\n// CGeneralSettingsDlg 消息处理程序\n\n\nBOOL CGeneralSettingsDlg::OnInitDialog()\n{\n    CTabDlg::OnInitDialog();\n\n    // TODO:  在此添加额外的初始化\n\n    ((CButton*)GetDlgItem(IDC_CHECK_UPDATE_CHECK))->SetCheck(m_data.check_update_when_start);\n    ((CButton*)GetDlgItem(IDC_ALLOW_SKIN_FONT_CHECK))->SetCheck(m_data.allow_skin_cover_font);\n    ((CButton*)GetDlgItem(IDC_ALLOW_SKIN_DISP_STR_CHECK))->SetCheck(m_data.allow_skin_cover_text);\n    if (theApp.IsForceShowNotifyIcon())\n    {\n        m_data.show_notify_icon = true;\n        EnableDlgCtrl(IDC_SHOW_NOTIFY_ICON_CHECK, FALSE);\n    }\n    CheckDlgButton(IDC_SHOW_NOTIFY_ICON_CHECK, m_data.show_notify_icon);\n\n    if (m_data.update_source == 0)\n        CheckDlgButton(IDC_GITHUB_RADIO, TRUE);\n    else\n        CheckDlgButton(IDC_GITEE_RADIO, TRUE);\n\n    //检查开始菜单的“启动”目录下有没有程序的快捷方式，如果有则设置开机自启动，然后删除快捷方式\n    wstring start_up_path = CCommon::GetStartUpPath();\n    bool shortcut_exist = CCommon::FileExist((start_up_path + L\"\\\\TrafficMonitor.lnk\").c_str());\n    if (shortcut_exist)\n    {\n        theApp.SetAutoRun(true);\n        m_data.auto_run = true;\n        DeleteFile((start_up_path + L\"\\\\TrafficMonitor.lnk\").c_str());\n    }\n    else\n    {\n        m_data.auto_run = theApp.GetAutoRun(&m_auto_run_path);\n    }\n\n    ((CButton*)GetDlgItem(IDC_SAVE_TO_APPDATA_RADIO))->SetCheck(!m_data.portable_mode);\n    ((CButton*)GetDlgItem(IDC_SAVE_TO_PROGRAM_DIR_RADIO))->SetCheck(m_data.portable_mode);\n    GetDlgItem(IDC_SAVE_TO_PROGRAM_DIR_RADIO)->EnableWindow(theApp.m_module_dir_writable);\n\n    ((CButton*)GetDlgItem(IDC_AUTO_RUN_CHECK))->SetCheck(m_data.auto_run);\n\n    ((CButton*)GetDlgItem(IDC_TODAY_TRAFFIC_TIP_CHECK))->SetCheck(m_data.traffic_tip_enable);\n    m_traffic_tip_edit.SetRange(1, 32767);\n    m_traffic_tip_edit.SetValue(m_data.traffic_tip_value);\n    m_traffic_tip_combo.AddString(_T(\"MB\"));\n    m_traffic_tip_combo.AddString(_T(\"GB\"));\n    m_traffic_tip_combo.SetCurSel(m_data.traffic_tip_unit);\n    CheckDlgButton(IDC_MEMORY_USAGE_TIP_CHECK, m_data.memory_usage_tip.enable);\n    m_memory_tip_edit.SetRange(1, 100);\n    m_memory_tip_edit.SetValue(m_data.memory_usage_tip.tip_value);\n\n    CheckDlgButton(IDC_CPU_TEMP_TIP_CHECK, m_data.cpu_temp_tip.enable);\n    m_cpu_temp_tip_edit.SetRange(1, 100);\n    m_cpu_temp_tip_edit.SetValue(m_data.cpu_temp_tip.tip_value);\n\n    CheckDlgButton(IDC_GPU_TEMP_TIP_CHECK, m_data.gpu_temp_tip.enable);\n    m_gpu_temp_tip_edit.SetRange(1, 100);\n    m_gpu_temp_tip_edit.SetValue(m_data.gpu_temp_tip.tip_value);\n\n    CheckDlgButton(IDC_HDD_TEMP_TIP_CHECK, m_data.hdd_temp_tip.enable);\n    m_hdd_temp_tip_edit.SetRange(1, 100);\n    m_hdd_temp_tip_edit.SetValue(m_data.hdd_temp_tip.tip_value);\n\n    CheckDlgButton(IDC_MBD_TEMP_TIP_CHECK, m_data.mainboard_temp_tip.enable);\n    m_mbd_temp_tip_edit.SetRange(1, 100);\n    m_mbd_temp_tip_edit.SetValue(m_data.mainboard_temp_tip.tip_value);\n\n    SetControlEnable();\n\n    m_language_combo.AddString(CCommon::LoadText(IDS_FOLLOWING_SYSTEM));\n    m_language_combo.AddString(_T(\"English\"));\n    m_language_combo.AddString(_T(\"简体中文\"));\n    m_language_combo.AddString(_T(\"繁體中文\"));\n    m_language_combo.SetCurSel(static_cast<int>(m_data.language));\n\n    ((CButton*)GetDlgItem(IDC_SHOW_ALL_CONNECTION_CHECK))->SetCheck(m_data.show_all_interface);\n\n    m_toolTip.Create(this);\n    m_toolTip.SetMaxTipWidth(theApp.DPI(300));\n    m_toolTip.AddTool(GetDlgItem(IDC_SHOW_ALL_CONNECTION_CHECK), CCommon::LoadText(IDS_SHOW_ALL_INFO_TIP));\n    m_toolTip.AddTool(GetDlgItem(IDC_SAVE_TO_APPDATA_RADIO), theApp.m_appdata_dir.c_str());\n    m_toolTip.AddTool(GetDlgItem(IDC_SAVE_TO_PROGRAM_DIR_RADIO), theApp.m_module_dir.c_str());\n    AddOrUpdateAutoRunTooltip(true);\n\n    ((CButton*)GetDlgItem(IDC_USE_CPU_TIME_RADIO))->SetCheck(m_data.m_get_cpu_usage_by_cpu_times);\n    ((CButton*)GetDlgItem(IDC_USE_PDH_RADIO))->SetCheck(!m_data.m_get_cpu_usage_by_cpu_times);\n\n    m_monitor_span_edit.SetRange(MONITOR_TIME_SPAN_MIN, MONITOR_TIME_SPAN_MAX);\n    m_monitor_span_edit.SetValue(m_data.monitor_time_span);\n\n    m_monitor_time_span_ori = m_data.monitor_time_span;\n    m_update_source_ori = m_data.update_source;\n\n#ifndef WITHOUT_TEMPERATURE\n    //初始化硬件监控Check box\n    CheckDlgButton(IDC_CPU_CHECK, m_data.IsHardwareEnable(HI_CPU));\n    CheckDlgButton(IDC_GPU_CHECK, m_data.IsHardwareEnable(HI_GPU));\n    CheckDlgButton(IDC_HDD_CHECK, m_data.IsHardwareEnable(HI_HDD));\n    CheckDlgButton(IDC_MBD_CHECK, m_data.IsHardwareEnable(HI_MBD));\n\n    if (theApp.m_pMonitor != nullptr)\n    {\n        CSingleLock sync(&theApp.m_minitor_lib_critical, TRUE);\n        //初始化选择硬盘下拉列表\n        for (const auto& hdd_item : theApp.m_pMonitor->AllHDDTemperature())\n            m_hard_disk_combo.AddString(hdd_item.first.c_str());\n        int cur_index = m_hard_disk_combo.FindString(-1, m_data.hard_disk_name.c_str());\n        m_hard_disk_combo.SetCurSel(cur_index);\n        //初始化选择CPU下拉列表\n        m_select_cpu_combo.AddString(CCommon::LoadText(IDS_AVREAGE_TEMPERATURE));\n        for (const auto& cpu_item : theApp.m_pMonitor->AllCpuTemperature())\n            m_select_cpu_combo.AddString(cpu_item.first.c_str());\n        cur_index = m_select_cpu_combo.FindString(-1, m_data.cpu_core_name.c_str());\n        if (cur_index < 0)\n            cur_index = 0;\n        m_select_cpu_combo.SetCurSel(cur_index);\n    }\n#endif\n\n    //不含温度监控的版本，禁用温度相关的控件\n#ifdef WITHOUT_TEMPERATURE\n    EnableDlgCtrl(IDC_CPU_TEMP_TIP_CHECK, false);\n    EnableDlgCtrl(IDC_CPU_TEMP_TIP_EDIT, false);\n    EnableDlgCtrl(IDC_GPU_TEMP_TIP_CHECK, false);\n    EnableDlgCtrl(IDC_GPU_TEMP_TIP_EDIT, false);\n    EnableDlgCtrl(IDC_HDD_TEMP_TIP_CHECK, false);\n    EnableDlgCtrl(IDC_HDD_TIP_EDIT, false);\n    EnableDlgCtrl(IDC_MBD_TEMP_TIP_CHECK, false);\n    EnableDlgCtrl(IDC_MBD_TEMP_TIP_EDIT, false);\n    EnableDlgCtrl(IDC_CPU_CHECK, false);\n    EnableDlgCtrl(IDC_GPU_CHECK, false);\n    EnableDlgCtrl(IDC_HDD_CHECK, false);\n    EnableDlgCtrl(IDC_MBD_CHECK, false);\n    EnableDlgCtrl(IDC_SELECT_HARD_DISK_COMBO, false);\n    EnableDlgCtrl(IDC_SELECT_CPU_COMBO, false);\n    EnableDlgCtrl(IDC_CPU_TEMP_STATIC, false);\n    EnableDlgCtrl(IDC_GPU_TEMP_STATIC, false);\n    EnableDlgCtrl(IDC_HDD_STATIC, false);\n    EnableDlgCtrl(IDC_MBD_TEMP_STATIC, false);\n    EnableDlgCtrl(IDC_SELECT_HDD_STATIC, false);\n    EnableDlgCtrl(IDC_SELECT_CPU_STATIC, false);\n    EnableDlgCtrl(IDC_HARDWARE_MONITOR_STATIC, false);\n#endif\n\n    m_plugin_manager_btn.SetIcon(theApp.GetMenuIcon(IDI_PLUGINS));\n    m_select_connection_btn.SetIcon(theApp.GetMenuIcon(IDI_CONNECTION));\n\n    return TRUE;  // return TRUE unless you set the focus to a control\n                  // 异常: OCX 属性页应返回 FALSE\n}\n\n\nvoid CGeneralSettingsDlg::OnBnClickedCheckNowButton()\n{\n    // TODO: 在此添加控件通知处理程序代码\n    theApp.CheckUpdateInThread(true);\n}\n\n\nvoid CGeneralSettingsDlg::OnBnClickedCheckUpdateCheck()\n{\n    // TODO: 在此添加控件通知处理程序代码\n    m_data.check_update_when_start = (((CButton*)GetDlgItem(IDC_CHECK_UPDATE_CHECK))->GetCheck() != 0);\n}\n\n\nvoid CGeneralSettingsDlg::OnBnClickedAutoRunCheck()\n{\n    // TODO: 在此添加控件通知处理程序代码\n    m_data.auto_run = (((CButton*)GetDlgItem(IDC_AUTO_RUN_CHECK))->GetCheck() != 0);\n    m_auto_run_modified = true;\n}\n\n\nvoid CGeneralSettingsDlg::OnBnClickedAllowSkinFontCheck()\n{\n    // TODO: 在此添加控件通知处理程序代码\n    m_data.allow_skin_cover_font = (((CButton*)GetDlgItem(IDC_ALLOW_SKIN_FONT_CHECK))->GetCheck() != 0);\n}\n\n\nvoid CGeneralSettingsDlg::OnBnClickedAllowSkinDispStrCheck()\n{\n    // TODO: 在此添加控件通知处理程序代码\n    m_data.allow_skin_cover_text = (((CButton*)GetDlgItem(IDC_ALLOW_SKIN_DISP_STR_CHECK))->GetCheck() != 0);\n}\n\n\nvoid CGeneralSettingsDlg::OnOK()\n{\n    // TODO: 在此添加专用代码和/或调用基类\n    //获取消息提示的设置\n    m_data.traffic_tip_value = m_traffic_tip_edit.GetValue();\n    if (m_data.traffic_tip_value < 1) m_data.traffic_tip_value = 1;\n    if (m_data.traffic_tip_value > 32767) m_data.traffic_tip_value = 32767;\n    m_data.traffic_tip_unit = m_traffic_tip_combo.GetCurSel();\n\n    auto checkTipValue = [](int& value)\n    {\n        if (value < 1) value = 1;\n        if (value > 100) value = 100;\n    };\n    m_data.memory_usage_tip.tip_value = m_memory_tip_edit.GetValue();\n    checkTipValue(m_data.memory_usage_tip.tip_value);\n\n    m_data.cpu_temp_tip.tip_value = m_cpu_temp_tip_edit.GetValue();\n    checkTipValue(m_data.cpu_temp_tip.tip_value);\n\n    m_data.gpu_temp_tip.tip_value = m_gpu_temp_tip_edit.GetValue();\n    checkTipValue(m_data.gpu_temp_tip.tip_value);\n\n    m_data.hdd_temp_tip.tip_value = m_hdd_temp_tip_edit.GetValue();\n    checkTipValue(m_data.hdd_temp_tip.tip_value);\n\n    m_data.mainboard_temp_tip.tip_value = m_mbd_temp_tip_edit.GetValue();\n    checkTipValue(m_data.mainboard_temp_tip.tip_value);\n\n    //获取语言的设置\n    m_data.language = static_cast<Language>(m_language_combo.GetCurSel());\n    if (m_data.language != theApp.m_general_data.language)\n    {\n        MessageBox(CCommon::LoadText(IDS_LANGUAGE_CHANGE_INFO), NULL, MB_ICONINFORMATION | MB_OK);\n    }\n    m_show_all_interface_modified = (m_data.show_all_interface != theApp.m_general_data.show_all_interface);\n\n    //获取数据文件保存位置的设置\n    m_data.portable_mode = (((CButton*)GetDlgItem(IDC_SAVE_TO_PROGRAM_DIR_RADIO))->GetCheck() != 0);\n    if (m_data.portable_mode != theApp.m_general_data.portable_mode)\n    {\n        MessageBox(CCommon::LoadText(IDS_CFG_DIR_CHANGED_INFO), NULL, MB_ICONINFORMATION | MB_OK);\n    }\n\n    m_data.monitor_time_span = m_monitor_span_edit.GetValue();\n\n    //m_taskbar_item_modified = (theApp.m_taskbar_data.m_tbar_display_item != taskbar_displat_item_ori);\n\n    CTabDlg::OnOK();\n}\n\n\nvoid CGeneralSettingsDlg::OnBnClickedTodayTrafficTipCheck()\n{\n    // TODO: 在此添加控件通知处理程序代码\n    m_data.traffic_tip_enable = (((CButton*)GetDlgItem(IDC_TODAY_TRAFFIC_TIP_CHECK))->GetCheck() != 0);\n    SetControlEnable();\n}\n\n\nvoid CGeneralSettingsDlg::OnBnClickedMemoryUsageTipCheck()\n{\n    // TODO: 在此添加控件通知处理程序代码\n    m_data.memory_usage_tip.enable = (((CButton*)GetDlgItem(IDC_MEMORY_USAGE_TIP_CHECK))->GetCheck() != 0);\n    SetControlEnable();\n}\n\n\nvoid CGeneralSettingsDlg::OnBnClickedOpenConfigPathButton()\n{\n    // TODO: 在此添加控件通知处理程序代码\n    ShellExecute(NULL, _T(\"explore\"), theApp.m_config_dir.c_str(), NULL, NULL, SW_SHOWNORMAL);\n}\n\n\nvoid CGeneralSettingsDlg::OnBnClickedShowAllConnectionCheck()\n{\n    // TODO: 在此添加控件通知处理程序代码\n    m_data.show_all_interface = (((CButton*)GetDlgItem(IDC_SHOW_ALL_CONNECTION_CHECK))->GetCheck() != 0);\n}\n\n\nBOOL CGeneralSettingsDlg::PreTranslateMessage(MSG* pMsg)\n{\n    // TODO: 在此添加专用代码和/或调用基类\n    if (pMsg->message == WM_MOUSEMOVE)\n        m_toolTip.RelayEvent(pMsg);\n\n    return CTabDlg::PreTranslateMessage(pMsg);\n}\n\n\nvoid CGeneralSettingsDlg::OnBnClickedUseCpuTimeRadio()\n{\n    // TODO: 在此添加控件通知处理程序代码\n    m_data.m_get_cpu_usage_by_cpu_times = true;\n}\n\n\nvoid CGeneralSettingsDlg::OnBnClickedUsePdhRadio()\n{\n    // TODO: 在此添加控件通知处理程序代码\n    m_data.m_get_cpu_usage_by_cpu_times = false;\n}\n\nvoid CGeneralSettingsDlg::OnDeltaposSpin(NMHDR* pNMHDR, LRESULT* pResult)\n{\n    //这里响应微调按钮（spin button）点击上下按钮时的事件，\n    //CSpinButtonCtrl的对象是作为CSpinEdit的成员变量的，而此消息会向CSpinButtonCtrl的父窗口发送，但是CSpinEdit不是它的父窗口，\n    //因此此消息无法在CSpinEdit中响应，只能在这里响应。\n    //所有CSpinEdit类中的Spin按钮点击时的响应都在这里，因为这些Spin按钮的ID都是“SPIN_ID”。\n    //通过GetBuddy的返回值判断微调按钮是属于哪个EditBox的。\n\n    CSpinButtonCtrl* pSpin = (CSpinButtonCtrl*)CWnd::FromHandle(pNMHDR->hwndFrom);\n    if (pSpin == nullptr)\n        return;\n    CWnd* pEdit = pSpin->GetBuddy();\n    if (pEdit == &m_monitor_span_edit)       //当用户点击了“监控时间间隔”的微调按钮时\n    {\n        LPNMUPDOWN pNMUpDown = reinterpret_cast<LPNMUPDOWN>(pNMHDR);\n        if (pNMUpDown->iDelta == -1)\n        {\n            // 用户按下了spin控件的向下箭头\n            int value = m_monitor_span_edit.GetValue();\n            value -= MONITOR_SPAN_STEP;\n            value /= MONITOR_SPAN_STEP;\n            value *= MONITOR_SPAN_STEP;\n            m_monitor_span_edit.SetValue(value);\n        }\n        else if (pNMUpDown->iDelta == 1)\n        {\n            // 用户按下了spin控件的向上箭头\n            int value = m_monitor_span_edit.GetValue();\n            value += MONITOR_SPAN_STEP;\n            value /= MONITOR_SPAN_STEP;\n            value *= MONITOR_SPAN_STEP;\n            m_monitor_span_edit.SetValue(value);\n        }\n        pNMUpDown->iDelta = 0;\n    }\n    *pResult = 0;\n}\n\n\n\nvoid CGeneralSettingsDlg::OnEnKillfocusMonitorSpanEdit()\n{\n    // TODO: 在此添加控件通知处理程序代码\n\n    //这里限制监控时间间隔只能输入100的倍数\n    CString str;\n    GetDlgItemText(IDC_MONITOR_SPAN_EDIT, str);\n    str.Replace(_T(\",\"), _T(\"\"));\n    int value = _ttoi(str.GetString());\n    if (value < MONITOR_TIME_SPAN_MIN || value > MONITOR_TIME_SPAN_MAX)\n    {\n        value = 1000;\n    }\n    else\n    {\n        value /= MONITOR_SPAN_STEP;\n        value *= MONITOR_SPAN_STEP;\n    }\n    m_monitor_span_edit.SetValue(value);\n}\n\n\nvoid CGeneralSettingsDlg::OnBnClickedCpuTempTipCheck()\n{\n    // TODO: 在此添加控件通知处理程序代码\n    m_data.cpu_temp_tip.enable = (IsDlgButtonChecked(IDC_CPU_TEMP_TIP_CHECK) != 0);\n    SetControlEnable();\n}\n\n\nvoid CGeneralSettingsDlg::OnBnClickedGpuTempTipCheck()\n{\n    // TODO: 在此添加控件通知处理程序代码\n    m_data.gpu_temp_tip.enable = (IsDlgButtonChecked(IDC_GPU_TEMP_TIP_CHECK) != 0);\n    SetControlEnable();\n}\n\n\nvoid CGeneralSettingsDlg::OnBnClickedHddTempTipCheck()\n{\n    // TODO: 在此添加控件通知处理程序代码\n    m_data.hdd_temp_tip.enable = (IsDlgButtonChecked(IDC_HDD_TEMP_TIP_CHECK) != 0);\n    SetControlEnable();\n}\n\n\nvoid CGeneralSettingsDlg::OnBnClickedMbdTempTipCheck()\n{\n    // TODO: 在此添加控件通知处理程序代码\n    m_data.mainboard_temp_tip.enable = (IsDlgButtonChecked(IDC_MBD_TEMP_TIP_CHECK) != 0);\n    SetControlEnable();\n}\n\n\nvoid CGeneralSettingsDlg::OnBnClickedGithubRadio()\n{\n    // TODO: 在此添加控件通知处理程序代码\n    m_data.update_source = 0;\n    theApp.m_general_data.update_source = 0;        //点击“更新源”的单选按钮时，同时更改theApp中的设置，以确保点击“立即检查”按钮时使用选择的更新源更新\n}\n\n\nvoid CGeneralSettingsDlg::OnBnClickedGiteeRadio()\n{\n    // TODO: 在此添加控件通知处理程序代码\n    m_data.update_source = 1;\n    theApp.m_general_data.update_source = 1;\n}\n\n\nvoid CGeneralSettingsDlg::OnCancel()\n{\n    // TODO: 在此添加专用代码和/或调用基类\n    theApp.m_general_data.update_source = m_update_source_ori;      //点击“取消”时恢复开始的“更新源”选项\n\n    CTabDlg::OnCancel();\n}\n\n\nvoid CGeneralSettingsDlg::OnBnClickedRestoreDefaultTimeSpanButton()\n{\n    // TODO: 在此添加控件通知处理程序代码\n    m_monitor_span_edit.SetValue(1000);\n}\n\n\nvoid CGeneralSettingsDlg::OnCbnSelchangeSelectHardDiskCombo()\n{\n    // TODO: 在此添加控件通知处理程序代码\n    CString hard_disk_name;\n    m_hard_disk_combo.GetWindowText(hard_disk_name);\n    m_data.hard_disk_name = hard_disk_name.GetString();\n}\n\n\nvoid CGeneralSettingsDlg::OnBnClickedCpuCheck()\n{\n    // TODO: 在此添加控件通知处理程序代码\n    bool checked = IsDlgButtonChecked(IDC_CPU_CHECK) != 0;\n    if (checked && !ShowHardwareMonitorWarning())\n    {\n        checked = false;\n        CheckDlgButton(IDC_CPU_CHECK, FALSE);\n    }\n    m_data.SetHardwareEnable(HI_CPU, checked);\n}\n\n\nvoid CGeneralSettingsDlg::OnBnClickedGpuCheck()\n{\n    // TODO: 在此添加控件通知处理程序代码\n    bool checked = IsDlgButtonChecked(IDC_GPU_CHECK) != 0;\n    if (checked && !ShowHardwareMonitorWarning())\n    {\n        checked = false;\n        CheckDlgButton(IDC_GPU_CHECK, FALSE);\n    }\n    m_data.SetHardwareEnable(HI_GPU, checked);\n}\n\n\nvoid CGeneralSettingsDlg::OnBnClickedHddCheck()\n{\n    // TODO: 在此添加控件通知处理程序代码\n    bool checked = IsDlgButtonChecked(IDC_HDD_CHECK) != 0;\n    if (checked && !ShowHardwareMonitorWarning())\n    {\n        checked = false;\n        CheckDlgButton(IDC_HDD_CHECK, FALSE);\n    }\n    m_data.SetHardwareEnable(HI_HDD, checked);\n}\n\n\nvoid CGeneralSettingsDlg::OnBnClickedMbdCheck()\n{\n    // TODO: 在此添加控件通知处理程序代码\n    bool checked = IsDlgButtonChecked(IDC_MBD_CHECK) != 0;\n    if (checked && !ShowHardwareMonitorWarning())\n    {\n        checked = false;\n        CheckDlgButton(IDC_MBD_CHECK, FALSE);\n    }\n    m_data.SetHardwareEnable(HI_MBD, checked);\n\n}\n\n\nvoid CGeneralSettingsDlg::OnCbnSelchangeSelectCpuCombo()\n{\n    // TODO: 在此添加控件通知处理程序代码\n    CString cpu_core_name;\n    m_select_cpu_combo.GetWindowText(cpu_core_name);\n    m_data.cpu_core_name = cpu_core_name.GetString();\n}\n\n\nvoid CGeneralSettingsDlg::OnBnClickedPluginManageButton()\n{\n    // TODO: 在此添加控件通知处理程序代码\n    CPluginManagerDlg dlg;\n    dlg.DoModal();\n}\n\n\nvoid CGeneralSettingsDlg::OnBnClickedShowNotifyIconCheck()\n{\n    // TODO: 在此添加控件通知处理程序代码\n    m_data.show_notify_icon = (IsDlgButtonChecked(IDC_SHOW_NOTIFY_ICON_CHECK) != 0);\n}\n\n\nvoid CGeneralSettingsDlg::OnBnClickedSelectConnectionsButton()\n{\n    // TODO: 在此添加控件通知处理程序代码\n    CSelectConnectionsDlg dlg(m_data.connections_hide);\n    if (dlg.DoModal() == IDOK)\n    {\n        m_data.connections_hide = dlg.GetData();\n    }\n}\n\n\nvoid CGeneralSettingsDlg::OnBnClickedResetAutoRunButton()\n{\n    //先删除开机自动运行\n    theApp.SetAutoRunByRegistry(false);\n    theApp.SetAutoRunByTaskScheduler(false);\n    if (!theApp.SetAutoRun(true))    //重新设置开机自动运行\n    {\n        MessageBox(CCommon::LoadText(IDS_SET_AUTO_RUN_FAILED_WARNING), NULL, MB_ICONWARNING | MB_OK);\n        return;\n    }\n    //获取开机自动运行的路径\n    bool auto_run = theApp.GetAutoRun(&m_auto_run_path);\n    //重新勾选“开机自动运行”复选框\n    CheckDlgButton(IDC_AUTO_RUN_CHECK, auto_run);\n    //更新鼠标提示\n    AddOrUpdateAutoRunTooltip(false);\n}\n"
  },
  {
    "path": "TrafficMonitor/GeneralSettingsDlg.h",
    "content": "﻿#pragma once\n#include \"TabDlg.h\"\n#include \"SpinEdit.h\"\n#include \"ComboBox2.h\"\n\n// CGeneralSettingsDlg dialog\n\nclass CGeneralSettingsDlg : public CTabDlg\n{\n    DECLARE_DYNAMIC(CGeneralSettingsDlg)\n\npublic:\n    CGeneralSettingsDlg(CWnd* pParent = NULL);   // standard constructor\n    virtual ~CGeneralSettingsDlg();\n\n    static void CheckTaskbarDisplayItem();\n\n    //选项设置数据\n    GeneralSettingData m_data;\n\n    // 对话框数据\n#ifdef AFX_DESIGN_TIME\n    enum { IDD = IDD_GENERAL_SETTINGS_DIALOG };\n#endif\n\npublic:\n    bool IsAutoRunModified() const { return m_auto_run_modified; }\n    bool IsShowAllInterfaceModified() const { return m_show_all_interface_modified; }\n    bool IsMonitorTimeSpanModified() const;\n    //bool IsTaskbarItemModified() const { return m_taskbar_item_modified; }\n\nprotected:\n    bool m_auto_run_modified{ false };      //如果更改了开机自动运行的设置，则会置为true\n    bool m_show_all_interface_modified{ false };\n    int m_monitor_time_span_ori{};\n    int m_update_source_ori{};\n    //bool m_taskbar_item_modified{ false };\n    wstring m_auto_run_path;\n\n    //控件变量\n    CSpinEdit m_traffic_tip_edit;\n    CComboBox2 m_traffic_tip_combo;\n    CSpinEdit m_memory_tip_edit;\n    CComboBox2 m_language_combo;\n    CToolTipCtrl m_toolTip;\n    CSpinEdit m_monitor_span_edit;\n    CSpinEdit m_cpu_temp_tip_edit;\n    CSpinEdit m_gpu_temp_tip_edit;\n    CSpinEdit m_hdd_temp_tip_edit;\n    CSpinEdit m_mbd_temp_tip_edit;\n    CComboBox2 m_hard_disk_combo;\n    CComboBox2 m_select_cpu_combo;\n    CButton m_plugin_manager_btn;\n    CButton m_select_connection_btn;\n\n    virtual void DoDataExchange(CDataExchange* pDX);    // DDX/DDV support\n\n    //设置控件的启用和禁用\n    void SetControlEnable();\n\n    virtual void SetControlMouseWheelEnable(bool enable) override;\n\n    //显示开启硬件监控时的提示，如果用户选择了“是”则返回true，否则返回false\n    //“以后不再显示该对话框”的标记保存在注册表“\\HKEY_CURRENT_USER\\SOFTWARE\\Microsoft\\Windows\\CurrentVersion\\Explorer\\DontShowMeThisDialogAgain”\n    bool ShowHardwareMonitorWarning();\n\n    //添加或更新开机自动运行的鼠标提示\n    void AddOrUpdateAutoRunTooltip(bool add);\n\n    DECLARE_MESSAGE_MAP()\npublic:\n    virtual BOOL OnInitDialog();\n    afx_msg void OnBnClickedCheckNowButton();\n    afx_msg void OnBnClickedCheckUpdateCheck();\n    afx_msg void OnBnClickedAutoRunCheck();\n    afx_msg void OnBnClickedAllowSkinFontCheck();\n    afx_msg void OnBnClickedAllowSkinDispStrCheck();\n    virtual void OnOK();\n    afx_msg void OnBnClickedTodayTrafficTipCheck();\n    afx_msg void OnBnClickedMemoryUsageTipCheck();\n    afx_msg void OnBnClickedOpenConfigPathButton();\n    afx_msg void OnBnClickedShowAllConnectionCheck();\n    virtual BOOL PreTranslateMessage(MSG* pMsg);\n    afx_msg void OnBnClickedUseCpuTimeRadio();\n    afx_msg void OnBnClickedUsePdhRadio();\n    afx_msg void OnDeltaposSpin(NMHDR* pNMHDR, LRESULT* pResult);\n    afx_msg void OnEnKillfocusMonitorSpanEdit();\n    afx_msg void OnBnClickedCpuTempTipCheck();\n    afx_msg void OnBnClickedGpuTempTipCheck();\n    afx_msg void OnBnClickedHddTempTipCheck();\n    afx_msg void OnBnClickedMbdTempTipCheck();\n    afx_msg void OnBnClickedGithubRadio();\n    afx_msg void OnBnClickedGiteeRadio();\n    virtual void OnCancel();\n    afx_msg void OnBnClickedRestoreDefaultTimeSpanButton();\n    afx_msg void OnCbnSelchangeSelectHardDiskCombo();\n    afx_msg void OnBnClickedCpuCheck();\n    afx_msg void OnBnClickedGpuCheck();\n    afx_msg void OnBnClickedHddCheck();\n    afx_msg void OnBnClickedMbdCheck();\n    afx_msg void OnCbnSelchangeSelectCpuCombo();\n    afx_msg void OnBnClickedPluginManageButton();\n    afx_msg void OnBnClickedShowNotifyIconCheck();\n    afx_msg void OnBnClickedSelectConnectionsButton();\n    afx_msg void OnBnClickedResetAutoRunButton();\n};\n"
  },
  {
    "path": "TrafficMonitor/HighResolutionTimer.h",
    "content": "﻿#include <MMSYSTEM.H>\n#pragma comment(lib, \"WINMM.LIB\")\n\nclass CHighResolutionTimer\n{\n    typedef void(*TIMERCALLBACK)(DWORD_PTR);\n\nprivate:\n    DWORD_PTR m_dwUser;\n    MMRESULT m_nIDTimer;\n    UINT m_uDelay;\n    TIMERCALLBACK m_pfnCallback;\n\npublic:\n    void KillTimer()\n    {\n        if (m_nIDTimer != NULL)\n        {\n            timeKillEvent(m_nIDTimer);\n            m_nIDTimer = NULL;\n        }\n    }\n\n    CHighResolutionTimer()\n    {\n        m_nIDTimer = NULL;\n        m_uDelay = 0;\n        m_pfnCallback = 0;\n    }\n\n    virtual ~CHighResolutionTimer()\n    {\n        KillTimer();\n    }\n\n    static void CALLBACK TimeProc(UINT uID, UINT uMsg, DWORD_PTR dwUser, DWORD_PTR dw1, DWORD_PTR dw2)\n    {\n        // dwUser contains ptr to Timer object\n        CHighResolutionTimer * ptimer = (CHighResolutionTimer *)dwUser;\n        // Call user-specified callback and pass back user specified data\n        (ptimer->m_pfnCallback) (ptimer->m_dwUser);\n    }\n\n    BOOL CreateTimer(DWORD_PTR dwUser, UINT uDelay, TIMERCALLBACK lpTimeProc)\n    {\n        ASSERT(dwUser);\n        ASSERT(lpTimeProc);\n\n        m_dwUser = dwUser;\n        m_pfnCallback = lpTimeProc;\n        BOOL ret = FALSE;\n        KillTimer();\n        m_nIDTimer = timeSetEvent(uDelay, 0, (LPTIMECALLBACK)TimeProc, (DWORD_PTR)this, TIME_PERIODIC | TIME_CALLBACK_FUNCTION);\n\n        if (m_nIDTimer != NULL)\n            ret = TRUE;\n        return(ret);\n    }\n};\n"
  },
  {
    "path": "TrafficMonitor/HistoryTrafficCalendarDlg.cpp",
    "content": "﻿// HistoryTrafficCalendarDlg.cpp: 实现文件\n//\n\n#include \"stdafx.h\"\n#include \"TrafficMonitor.h\"\n#include \"HistoryTrafficCalendarDlg.h\"\n#include \"afxdialogex.h\"\n\n\n// CHistoryTrafficCalendarDlg 对话框\n\nIMPLEMENT_DYNAMIC(CHistoryTrafficCalendarDlg, CTabDlg)\n\nCHistoryTrafficCalendarDlg::CHistoryTrafficCalendarDlg(deque<HistoryTraffic>& history_traffics, CWnd* pParent /*=nullptr*/)\n    : CTabDlg(IDD_HISTORY_TRAFFIC_CALENDAR_DIALOG, pParent), m_history_traffics(history_traffics)\n{\n\n}\n\nCHistoryTrafficCalendarDlg::~CHistoryTrafficCalendarDlg()\n{\n}\n\nvoid CHistoryTrafficCalendarDlg::SetDayTraffic()\n{\n    for (int i{}; i < CALENDAR_HEIGHT; i++)\n    {\n        for (int j{}; j < CALENDAR_WIDTH; j++)\n        {\n            HistoryTraffic history_traffic;\n            history_traffic.year = m_year;\n            history_traffic.month = m_month;\n            history_traffic.day = m_calendar[i][j].day;\n            //使用二分法查找日历中每一个日期的流量\n            if (history_traffic.day > 0)\n            {\n                if (std::binary_search(m_history_traffics.begin(), m_history_traffics.end(), history_traffic, HistoryTraffic::DateGreater))\n                {\n                    auto iter = std::lower_bound(m_history_traffics.begin(), m_history_traffics.end(), history_traffic, HistoryTraffic::DateGreater);\n                    if (iter != m_history_traffics.end())\n                    {\n                        m_calendar[i][j].up_traffic = iter->up_kBytes;\n                        m_calendar[i][j].down_traffic = iter->down_kBytes;\n                        m_calendar[i][j].mixed = iter->mixed;\n\n                    }\n                }\n            }\n        }\n    }\n}\n\nvoid CHistoryTrafficCalendarDlg::MonthSelectChanged()\n{\n    CCalendarHelper::GetCalendar(m_year, m_month, m_calendar, theApp.m_cfg_data.m_sunday_first);\n    SetDayTraffic();\n    CalculateMonthTotalTraffic();\n    InvalidateRect(m_draw_rect);\n}\n\nvoid CHistoryTrafficCalendarDlg::CalculateMonthTotalTraffic()\n{\n    m_month_total_upload = 0;\n    m_month_total_download = 0;\n    for (int i{}; i < CALENDAR_HEIGHT; i++)\n    {\n        for (int j{}; j < CALENDAR_WIDTH; j++)\n        {\n            m_month_total_upload += m_calendar[i][j].up_traffic;\n            m_month_total_download += m_calendar[i][j].down_traffic;\n        }\n    }\n}\n\nvoid CHistoryTrafficCalendarDlg::SetComboSel()\n{\n    int cnt{};\n    int year_selected;\n    for (int i{ m_year_max }; i >= m_year_min; i--)\n    {\n        if (i == m_year)\n            year_selected = cnt;\n        cnt++;\n    }\n    m_year_combo.SetCurSel(year_selected);\n    m_month_combo.SetCurSel(m_month - 1);\n}\n\nvoid CHistoryTrafficCalendarDlg::DoDataExchange(CDataExchange* pDX)\n{\n    CTabDlg::DoDataExchange(pDX);\n    DDX_Control(pDX, IDC_YEAR_COMBO, m_year_combo);\n    DDX_Control(pDX, IDC_MONTH_COMBO, m_month_combo);\n}\n\n\nbool CHistoryTrafficCalendarDlg::IsWeekend(int index)\n{\n    if (theApp.m_cfg_data.m_sunday_first)\n        return (index == 0 || index == 6);\n    else\n        return (index == 5 || index == 6);\n}\n\nCString CHistoryTrafficCalendarDlg::GetWeekdayString(int index)\n{\n    CString str;\n    if (!theApp.m_cfg_data.m_sunday_first)\n    {\n        index++;\n        if (index > 6)\n            index = 0;\n    }\n    switch (index)\n    {\n    case 0:\n        str.LoadString(IDS_SUNDAY);\n        break;\n    case 1:\n        str.LoadString(IDS_MONDAY);\n        break;\n    case 2:\n        str.LoadString(IDS_TUESDAY);\n        break;\n    case 3:\n        str.LoadString(IDS_WEDNESDAY);\n        break;\n    case 4:\n        str.LoadString(IDS_THURSDAY);\n        break;\n    case 5:\n        str.LoadString(IDS_FRIDAY);\n        break;\n    case 6:\n        str.LoadString(IDS_SATURDAY);\n        break;\n    }\n    return str;\n}\n\nBEGIN_MESSAGE_MAP(CHistoryTrafficCalendarDlg, CTabDlg)\n    ON_WM_PAINT()\n    ON_CBN_SELCHANGE(IDC_YEAR_COMBO, &CHistoryTrafficCalendarDlg::OnCbnSelchangeYearCombo)\n    ON_CBN_SELCHANGE(IDC_MONTH_COMBO, &CHistoryTrafficCalendarDlg::OnCbnSelchangeMonthCombo)\n    ON_WM_MOUSEMOVE()\n    ON_BN_CLICKED(IDC_PREVIOUS_BUTTON, &CHistoryTrafficCalendarDlg::OnBnClickedPreviousButton)\n    ON_BN_CLICKED(IDC_NEXT_BUTTON, &CHistoryTrafficCalendarDlg::OnBnClickedNextButton)\n    ON_WM_MOUSEWHEEL()\n    ON_BN_CLICKED(IDC_MENU_BUTTON, &CHistoryTrafficCalendarDlg::OnBnClickedMenuButton)\n    ON_WM_INITMENU()\n    ON_COMMAND(ID_FIRST_DAY_OF_WEEK_SUNDAY, &CHistoryTrafficCalendarDlg::OnFirstDayOfWeekSunday)\n    ON_COMMAND(ID_FIRST_DAY_OF_WEEK_MONDAY, &CHistoryTrafficCalendarDlg::OnFirstDayOfWeekMonday)\n    ON_COMMAND(ID_CALENDAR_JUMP_TO_TODAY, &CHistoryTrafficCalendarDlg::OnCalendarJumpToToday)\nEND_MESSAGE_MAP()\n\n\n// CHistoryTrafficCalendarDlg 消息处理程序\n\n\nBOOL CHistoryTrafficCalendarDlg::OnInitDialog()\n{\n    CTabDlg::OnInitDialog();\n\n    // TODO:  在此添加额外的初始化\n    m_year = m_history_traffics[0].year;\n    m_month = m_history_traffics[0].month;\n    CCalendarHelper::GetCalendar(m_year, m_month, m_calendar, theApp.m_cfg_data.m_sunday_first);\n    SetDayTraffic();\n    CalculateMonthTotalTraffic();\n\n    //初始化Combo Box\n    m_year_max = m_history_traffics[0].year;\n    m_year_min = m_history_traffics.back().year;\n    for (int i{ m_year_max }; i >= m_year_min; i--)\n    {\n        m_year_combo.AddString(CCommon::IntToString(i));\n    }\n    m_year_combo.SetCurSel(0);\n    for (int i{ 1 }; i <= 12; i++)\n    {\n        m_month_combo.AddString(CCommon::IntToString(i));\n    }\n    m_month_combo.SetCurSel(m_month - 1);\n\n    //初始化鼠标提示\n    m_tool_tips.Create(this, TTS_ALWAYSTIP | TTS_NOPREFIX);\n    m_tool_tips.SetMaxTipWidth(800);        //为鼠标提示设置一个最大宽度，以允许其换行\n    m_tool_tips.AddTool(this, _T(\"\"));\n\n    m_menu.LoadMenu(IDR_HISTORY_TRAFFIC_MENU);\n\n    return TRUE;  // return TRUE unless you set the focus to a control\n                  // 异常: OCX 属性页应返回 FALSE\n}\n\n\nvoid CHistoryTrafficCalendarDlg::OnPaint()\n{\n    CPaintDC dc(this); // device context for painting\n                       // TODO: 在此处添加消息处理程序代码\n                       // 不为绘图消息调用 CTabDlg::OnPaint()\n    CRect wndRect;\n    GetWindowRect(wndRect);\n    const int width = theApp.DPI(40);       //日历中每一个格子的宽度\n    const int height = theApp.DPI(30);      //日历中第一个格子的高度\n    m_draw_rect.left = m_start_x;\n    m_draw_rect.top = m_start_y;\n    //m_draw_rect.right = m_draw_rect.left + (CALENDAR_WIDTH*width);\n    m_draw_rect.right = wndRect.Width() - m_start_x;\n    m_draw_rect.bottom = m_draw_rect.top + ((CALENDAR_HEIGHT + 2) * height) + theApp.DPI(20);\n\n    //使用双缓冲绘图\n    CDrawDoubleBuffer draw_double_buffer(&dc, m_draw_rect);\n\n    //绘图\n    CDrawCommon draw;\n    draw.Create(draw_double_buffer.GetMemDC(), this);\n    draw.FillRect(CRect(0, 0, m_draw_rect.Width(), m_draw_rect.Height()), RGB(255, 255, 255));      //填充白色背景色\n    CRect rect{};\n    rect.left = 0;\n    rect.top = 0;\n    rect.right = rect.left + width;\n    rect.bottom = rect.top + height;\n    //画星期的行\n    for (int i{}; i < CALENDAR_WIDTH; i++)\n    {\n        rect.MoveToX(i * width);\n\n        if (IsWeekend(i))\n            draw.SetBackColor(RGB(217, 86, 86));\n        else if (i % 2 == 0)\n            draw.SetBackColor(RGB(1, 84, 151));\n        else\n            draw.SetBackColor(RGB(1, 107, 191));\n        CString str = GetWeekdayString(i);\n        draw.DrawWindowText(rect, str, RGB(255, 255, 255), Alignment::CENTER, true);\n    }\n\n    //绘制日历\n    for (int i{}; i < CALENDAR_HEIGHT; i++)\n    {\n        for (int j{}; j < CALENDAR_WIDTH; j++)\n        {\n            //设置日历中每一天的矩形的位置\n            rect.MoveToXY(j * width, height + theApp.DPI(2) + i * height);\n            //保存矩形的位置\n            m_calendar[i][j].rect = rect;\n            //绘制日期的数据\n            //绘制格子的背景颜色\n            if (IsWeekend(j))       //是周末时\n            {\n                if ((i + j) % 2 == 0)\n                    draw.SetBackColor(RGB(250, 234, 234));\n                else\n                    draw.SetBackColor(RGB(252, 242, 242));\n            }\n            else\n            {\n                if ((i + j) % 2 == 0)\n                    draw.SetBackColor(RGB(226, 241, 254));\n                else\n                    draw.SetBackColor(RGB(236, 246, 254));\n            }\n            draw.FillRectWithBackColor(rect);\n            //绘制格子上的日期的数字\n            CRect day_rect{ rect };\n            day_rect.bottom -= (rect.Height() / 2);\n            COLORREF text_color;\n            if (IsWeekend(j))\n                text_color = RGB(131, 29, 28);\n            else\n                text_color = RGB(0, 57, 107);\n            if (m_calendar[i][j].day != 0)\n                draw.DrawWindowText(day_rect, CCommon::IntToString(m_calendar[i][j].day), text_color, Alignment::CENTER, true);\n            //在今天的日期上画一个矩形框\n            COLORREF frame_color;\n            if (IsWeekend(j))\n                frame_color = RGB(218, 91, 91);\n            else\n                frame_color = RGB(1, 133, 238);\n            if (m_year == m_history_traffics[0].year && m_month == m_history_traffics[0].month && m_calendar[i][j].day == m_history_traffics[0].day)\n                draw.DrawRectOutLine(rect, frame_color, theApp.DPI(2));\n\n            //绘制指示流量大小的矩形\n            COLORREF color;\n            if (m_calendar[i][j].traffic() < 1024 * 1024)       //流量小于1GB时绘制蓝色\n                color = TRAFFIC_COLOR_BLUE;\n            else if (m_calendar[i][j].traffic() < 10 * 1024 * 1024) //流量小于10GB时绘制绿色\n                color = TRAFFIC_COLOR_GREEN;\n            else if (m_calendar[i][j].traffic() < 100 * 1024 * 1024)    //流量小于100GB时绘制黄色\n                color = TRAFFIC_COLOR_YELLOE;\n            else if (m_calendar[i][j].traffic() < 1024 * 1024 * 1024)   //流量小于1TB时绘制红色\n                color = TRAFFIC_COLOR_RED;\n            else        //流量超过1TB时显示深红色\n                color = TRAFFIC_COLOR_DARK_RED;\n            if (m_calendar[i][j].traffic() > 0)\n            {\n                CRect traffic_rect;\n                traffic_rect.left = rect.left + theApp.DPI(14);\n                traffic_rect.right = traffic_rect.left + theApp.DPI(12);\n                traffic_rect.top = rect.top + theApp.DPI(16);\n                traffic_rect.bottom = traffic_rect.top + theApp.DPI(12);\n                draw.FillRect(traffic_rect, color);\n            }\n        }\n    }\n\n    const COLORREF text_color{ RGB(0, 57, 107) };\n    //画当前月总流量\n    CString info;\n    info.Format(_T(\"%s %s (%s: %s, %s: %s)\"), CCommon::LoadText(IDS_CURRENT_MONTH_TOTAL_TRAFFIC),\n        CCommon::KBytesToString(m_month_total_upload + m_month_total_download),\n        CCommon::LoadText(IDS_UPLOAD), CCommon::KBytesToString(m_month_total_upload),\n        CCommon::LoadText(IDS_DOWNLOAD), CCommon::KBytesToString(m_month_total_download)\n    );\n    CRect info_rect;\n    info_rect.left = 0;\n    info_rect.top = height * (CALENDAR_HEIGHT + 1) + theApp.DPI(5);\n    info_rect.right = info_rect.left + m_draw_rect.Width();\n    info_rect.bottom = info_rect.top + theApp.DPI(36);\n    draw.SetBackColor(RGB(255, 255, 255));\n    draw.DrawWindowText(info_rect, info, text_color, Alignment::LEFT, true, true);\n\n    //画图例\n    CRect rc_legend{ info_rect };\n    rc_legend.MoveToXY(CALENDAR_WIDTH * width + theApp.DPI(32), theApp.DPI(16));\n    rc_legend.right = m_draw_rect.Width();\n    rc_legend.bottom = rc_legend.top + theApp.DPI(16);\n\n    draw.DrawWindowText(rc_legend, CCommon::LoadText(IDS_LEGEND, _T(\":\")), text_color, Alignment::LEFT, true, true);\n    rc_legend.MoveToY(rc_legend.bottom + theApp.DPI(6));\n    CRect rc_legend_box{ rc_legend };\n    const int box_side{ theApp.DPI(12) };\n    const int line_gap{ theApp.DPI(6) };\n    rc_legend_box.top = rc_legend.top + (rc_legend.Height() - box_side) / 2;\n    rc_legend_box.bottom = rc_legend_box.top + box_side;\n    rc_legend_box.right = rc_legend_box.left + box_side;\n    rc_legend.left = rc_legend_box.right + theApp.DPI(4);\n    //蓝色图例\n    draw.FillRect(rc_legend_box, TRAFFIC_COLOR_BLUE);\n    draw.DrawWindowText(rc_legend, _T(\"0~1GB\"), text_color);\n    //绿色图例\n    rc_legend.MoveToY(rc_legend.bottom + line_gap);\n    rc_legend_box.MoveToY(rc_legend.top + (rc_legend.Height() - box_side) / 2);\n    draw.FillRect(rc_legend_box, TRAFFIC_COLOR_GREEN);\n    draw.DrawWindowText(rc_legend, _T(\"1GB~10GB\"), text_color);\n    //黄色图例\n    rc_legend.MoveToY(rc_legend.bottom + line_gap);\n    rc_legend_box.MoveToY(rc_legend.top + (rc_legend.Height() - box_side) / 2);\n    draw.FillRect(rc_legend_box, TRAFFIC_COLOR_YELLOE);\n    draw.DrawWindowText(rc_legend, _T(\"10GB~100GB\"), text_color);\n    //红色图例\n    rc_legend.MoveToY(rc_legend.bottom + line_gap);\n    rc_legend_box.MoveToY(rc_legend.top + (rc_legend.Height() - box_side) / 2);\n    draw.FillRect(rc_legend_box, TRAFFIC_COLOR_RED);\n    draw.DrawWindowText(rc_legend, _T(\"100GB~1TB\"), text_color);\n    //深红色图例\n    rc_legend.MoveToY(rc_legend.bottom + line_gap);\n    rc_legend_box.MoveToY(rc_legend.top + (rc_legend.Height() - box_side) / 2);\n    draw.FillRect(rc_legend_box, TRAFFIC_COLOR_DARK_RED);\n    draw.DrawWindowText(rc_legend, _T(\"1TB~\"), text_color);\n}\n\n\nvoid CHistoryTrafficCalendarDlg::OnCbnSelchangeYearCombo()\n{\n    // TODO: 在此添加控件通知处理程序代码\n    int index = m_year_combo.GetCurSel();\n    CString str;\n    m_year_combo.GetLBText(index, str);\n    m_year = _ttoi(str);\n    MonthSelectChanged();\n}\n\n\nvoid CHistoryTrafficCalendarDlg::OnCbnSelchangeMonthCombo()\n{\n    // TODO: 在此添加控件通知处理程序代码\n    m_month = m_month_combo.GetCurSel() + 1;\n    MonthSelectChanged();\n}\n\n\nvoid CHistoryTrafficCalendarDlg::OnMouseMove(UINT nFlags, CPoint point)\n{\n    // TODO: 在此添加消息处理程序代码和/或调用默认值\n    int tip_day{};      //要显示鼠标提示的日期\n    DayTraffic tip_traffic{};\n    static int last_tip_day{};\n    //查找鼠标指针在哪个日期的矩形内\n    for (int i{}; i < CALENDAR_HEIGHT; i++)\n    {\n        for (int j{}; j < CALENDAR_WIDTH; j++)\n        {\n            CRect rect{ m_calendar[i][j].rect };\n            rect.OffsetRect(m_start_x, m_start_y);\n            if (rect.PtInRect(point))\n            {\n                tip_day = m_calendar[i][j].day;\n                tip_traffic = m_calendar[i][j];\n            }\n        }\n    }\n    bool show_tip = (tip_day > 0);\n    if (show_tip && last_tip_day != tip_day)\n    {\n        CString tip_info;\n        tip_info.Format(_T(\"%d/%d/%d\\r\\n\"), m_year, m_month, tip_day);\n        tip_info += CCommon::LoadText(IDS_TRAFFIC_USED1);\n        tip_info += CCommon::KBytesToString(tip_traffic.traffic());\n        if (!tip_traffic.mixed && tip_traffic.traffic() > 0)\n        {\n            tip_info += _T(\"\\r\\n\");\n            tip_info += CCommon::LoadText(IDS_UPLOAD, _T(\": \"));\n            tip_info += CCommon::KBytesToString(tip_traffic.up_traffic);\n            tip_info += _T(\"\\r\\n\");\n            tip_info += CCommon::LoadText(IDS_DOWNLOAD, _T(\": \"));\n            tip_info += CCommon::KBytesToString(tip_traffic.down_traffic);\n        }\n\n        m_tool_tips.AddTool(this, tip_info);\n        m_tool_tips.Pop();\n        last_tip_day = tip_day;\n    }\n    if (!show_tip)\n    {\n        m_tool_tips.AddTool(this, _T(\"\"));\n        m_tool_tips.Pop();\n        last_tip_day = 0;\n    }\n\n    CTabDlg::OnMouseMove(nFlags, point);\n}\n\n\nBOOL CHistoryTrafficCalendarDlg::PreTranslateMessage(MSG* pMsg)\n{\n    // TODO: 在此添加专用代码和/或调用基类\n    if (m_tool_tips.GetSafeHwnd() != 0)\n    {\n        m_tool_tips.RelayEvent(pMsg);\n    }\n\n    if (pMsg->message == WM_KEYDOWN)\n    {\n        if (pMsg->wParam == VK_LEFT)\n        {\n            OnBnClickedPreviousButton();\n            return TRUE;\n        }\n        if (pMsg->wParam == VK_RIGHT)\n        {\n            OnBnClickedNextButton();\n            return TRUE;\n        }\n    }\n\n    return CTabDlg::PreTranslateMessage(pMsg);\n}\n\n\nvoid CHistoryTrafficCalendarDlg::OnBnClickedPreviousButton()\n{\n    // TODO: 在此添加控件通知处理程序代码\n    if (m_year == m_year_min && m_month == 1)\n        return;\n    m_month--;\n    if (m_month <= 0)\n    {\n        m_month = 12;\n        m_year--;\n    }\n    SetComboSel();\n    MonthSelectChanged();\n}\n\n\nvoid CHistoryTrafficCalendarDlg::OnBnClickedNextButton()\n{\n    // TODO: 在此添加控件通知处理程序代码\n    if (m_year == m_year_max && m_month == 12)\n        return;\n    m_month++;\n    if (m_month > 12)\n    {\n        m_month = 1;\n        m_year++;\n    }\n    SetComboSel();\n    MonthSelectChanged();\n}\n\n\nBOOL CHistoryTrafficCalendarDlg::OnMouseWheel(UINT nFlags, short zDelta, CPoint pt)\n{\n    // TODO: 在此添加消息处理程序代码和/或调用默认值\n    //通过鼠标滚轮翻页\n    if (zDelta > 0)\n    {\n        OnBnClickedPreviousButton();\n    }\n    if (zDelta < 0)\n    {\n        OnBnClickedNextButton();\n    }\n\n    return CTabDlg::OnMouseWheel(nFlags, zDelta, pt);\n}\n\n\nvoid CHistoryTrafficCalendarDlg::OnBnClickedMenuButton()\n{\n    // TODO: 在此添加控件通知处理程序代码\n    CWnd* pBtn = GetDlgItem(IDC_MENU_BUTTON);\n    if (pBtn != nullptr)\n    {\n        CRect rect;\n        pBtn->GetWindowRect(rect);\n        CPoint point;\n        point.x = rect.left;\n        point.y = rect.bottom;\n        CMenu* pMenu = m_menu.GetSubMenu(1);\n        if (pMenu != NULL)\n            pMenu->TrackPopupMenu(TPM_LEFTALIGN | TPM_RIGHTBUTTON, point.x, point.y, this);\n    }\n}\n\n\nvoid CHistoryTrafficCalendarDlg::OnInitMenu(CMenu* pMenu)\n{\n    CTabDlg::OnInitMenu(pMenu);\n\n    // TODO: 在此处添加消息处理程序代码\n    if (theApp.m_cfg_data.m_sunday_first)\n        pMenu->CheckMenuRadioItem(ID_FIRST_DAY_OF_WEEK_SUNDAY, ID_FIRST_DAY_OF_WEEK_MONDAY, ID_FIRST_DAY_OF_WEEK_SUNDAY, MF_BYCOMMAND | MF_CHECKED);\n    else\n        pMenu->CheckMenuRadioItem(ID_FIRST_DAY_OF_WEEK_SUNDAY, ID_FIRST_DAY_OF_WEEK_MONDAY, ID_FIRST_DAY_OF_WEEK_MONDAY, MF_BYCOMMAND | MF_CHECKED);\n\n}\n\n\nvoid CHistoryTrafficCalendarDlg::OnFirstDayOfWeekSunday()\n{\n    // TODO: 在此添加命令处理程序代码\n    theApp.m_cfg_data.m_sunday_first = true;\n    MonthSelectChanged();\n}\n\n\nvoid CHistoryTrafficCalendarDlg::OnFirstDayOfWeekMonday()\n{\n    // TODO: 在此添加命令处理程序代码\n    theApp.m_cfg_data.m_sunday_first = false;\n    MonthSelectChanged();\n}\n\n\nvoid CHistoryTrafficCalendarDlg::OnCalendarJumpToToday()\n{\n    // TODO: 在此添加命令处理程序代码\n    m_year = m_history_traffics[0].year;\n    m_month = m_history_traffics[0].month;\n    SetComboSel();\n    MonthSelectChanged();\n\n}\n"
  },
  {
    "path": "TrafficMonitor/HistoryTrafficDlg.cpp",
    "content": "﻿// HistoryTrafficDlg.cpp : 实现文件\n//\n\n#include \"stdafx.h\"\n#include \"TrafficMonitor.h\"\n#include \"HistoryTrafficDlg.h\"\n#include \"afxdialogex.h\"\n\n\n// CHistoryTrafficDlg 对话框\n\nIMPLEMENT_DYNAMIC(CHistoryTrafficDlg, CBaseDialog)\n\nCHistoryTrafficDlg::CHistoryTrafficDlg(deque<HistoryTraffic>& history_traffics, CWnd* pParent /*=NULL*/)\n    : CBaseDialog(IDD_HISTORY_TRAFFIC_DIALOG, pParent), m_history_traffics(history_traffics), m_tab1_dlg(history_traffics, this), m_tab2_dlg(history_traffics, this)\n{\n\n}\n\nCHistoryTrafficDlg::~CHistoryTrafficDlg()\n{\n}\n\n\nCString CHistoryTrafficDlg::GetDialogName() const\n{\n    return _T(\"HistoryTrafficDlg\");\n}\n\nvoid CHistoryTrafficDlg::DoDataExchange(CDataExchange* pDX)\n{\n    CBaseDialog::DoDataExchange(pDX);\n    DDX_Control(pDX, IDC_TAB1, m_tab);\n}\n\nvoid CHistoryTrafficDlg::SetTabWndSize()\n{\n    CRect rect;\n    m_tab.GetClientRect(rect);\n    CRect rcTabItem;\n    m_tab.GetItemRect(0, rcTabItem);\n    rect.top += rcTabItem.Height() + 4;\n    rect.left += 4;\n    rect.bottom -= 4;\n    rect.right -= 4;\n    m_tab1_dlg.MoveWindow(&rect);\n    m_tab2_dlg.MoveWindow(&rect);\n}\n\n\nBEGIN_MESSAGE_MAP(CHistoryTrafficDlg, CBaseDialog)\n    ON_WM_GETMINMAXINFO()\n    ON_NOTIFY(TCN_SELCHANGE, IDC_TAB1, &CHistoryTrafficDlg::OnTcnSelchangeTab1)\n    ON_WM_SIZE()\nEND_MESSAGE_MAP()\n\n\n// CHistoryTrafficDlg 消息处理程序\n\n\nBOOL CHistoryTrafficDlg::OnInitDialog()\n{\n    CBaseDialog::OnInitDialog();\n\n    // TODO:  在此添加额外的初始化\n    SetWindowText(CCommon::LoadText(IDS_TITLE_HISTORY_TRAFFIC));\n    SetIcon(theApp.GetMenuIcon(IDI_STATISTICS), FALSE);\t\t// 设置小图标\n\n    //插入标签\n    m_tab.InsertItem(0, CCommon::LoadText(IDS_LIST_VIEW));\n    m_tab.InsertItem(1, CCommon::LoadText(IDS_CALENDAR_VIEW));\n    //创建子对话框\n    m_tab1_dlg.Create(IDD_HISTORY_TRAFFIC_LIST_DIALOG, &m_tab);\n    m_tab2_dlg.Create(IDD_HISTORY_TRAFFIC_CALENDAR_DIALOG, &m_tab);\n    //调整子对话框的大小和位置\n    SetTabWndSize();\n    //设置默认选中的标签\n    switch (m_tab_selected)\n    {\n    case 0:\n        m_tab1_dlg.ShowWindow(SW_SHOW);\n        break;\n    case 1:\n        m_tab2_dlg.ShowWindow(SW_SHOW);\n        break;\n    }\n    m_tab.SetCurFocus(m_tab_selected);\n\n    return TRUE;  // return TRUE unless you set the focus to a control\n                  // 异常: OCX 属性页应返回 FALSE\n}\n\n\n\nBOOL CHistoryTrafficDlg::PreTranslateMessage(MSG* pMsg)\n{\n    // TODO: 在此添加专用代码和/或调用基类\n    //if (GetKeyState(VK_CONTROL) & 0x80)\n    //{\n    //\tif (pMsg->wParam == 'D')\n    //\t{\n    //\t\tHistoryTraffic h{};\n    //\t\th.year = 2018;\n    //\t\th.month = 4;\n    //\t\th.day = 29;\n    //\t\tauto iter = std::lower_bound(m_history_traffics.begin(), m_history_traffics.end(), h, HistoryTraffic::DateGreater);\n    //\t\tint index = iter - m_history_traffics.begin();\n    //\t}\n    //}\n\n    return CBaseDialog::PreTranslateMessage(pMsg);\n}\n\nvoid CHistoryTrafficDlg::OnTcnSelchangeTab1(NMHDR* pNMHDR, LRESULT* pResult)\n{\n    // TODO: 在此添加控件通知处理程序代码\n    m_tab_selected = m_tab.GetCurSel();\n    switch (m_tab_selected)\n    {\n    case 0:\n        m_tab1_dlg.ShowWindow(SW_SHOW);\n        m_tab2_dlg.ShowWindow(SW_HIDE);\n        m_tab1_dlg.SetFocus();\n        break;\n    case 1:\n        m_tab2_dlg.ShowWindow(SW_SHOW);\n        m_tab1_dlg.ShowWindow(SW_HIDE);\n        m_tab2_dlg.SetFocus();\n        break;\n    }\n    *pResult = 0;\n}\n\n\nvoid CHistoryTrafficDlg::OnSize(UINT nType, int cx, int cy)\n{\n    CBaseDialog::OnSize(nType, cx, cy);\n\n    // TODO: 在此处添加消息处理程序代码\n    if (nType != SIZE_MINIMIZED && m_tab1_dlg.GetSafeHwnd() != NULL && m_tab2_dlg.GetSafeHwnd() != NULL)\n    {\n        SetTabWndSize();\n    }\n}\n"
  },
  {
    "path": "TrafficMonitor/HistoryTrafficDlg.h",
    "content": "﻿#pragma once\n#include \"afxcmn.h\"\n#include \"Common.h\"\n#include \"HistoryTrafficListDlg.h\"\n#include \"HistoryTrafficCalendarDlg.h\"\n#include \"BaseDialog.h\"\n\n// CHistoryTrafficDlg 对话框\n\nclass CHistoryTrafficDlg : public CBaseDialog\n{\n    DECLARE_DYNAMIC(CHistoryTrafficDlg)\n\npublic:\n    CHistoryTrafficDlg(deque<HistoryTraffic>& history_traffics, CWnd* pParent = NULL);   // 标准构造函数\n    virtual ~CHistoryTrafficDlg();\n\n    // 对话框数据\n#ifdef AFX_DESIGN_TIME\n    enum { IDD = IDD_HISTORY_TRAFFIC_DIALOG };\n#endif\n\npublic:\n    CHistoryTrafficListDlg m_tab1_dlg;\n    CHistoryTrafficCalendarDlg m_tab2_dlg;\n\nprotected:\n    deque<HistoryTraffic>& m_history_traffics;\n\n    CTabCtrl m_tab;\n    int m_tab_selected;\n\nprotected:\n    virtual void DoDataExchange(CDataExchange* pDX);    // DDX/DDV 支持\n    void SetTabWndSize();\n    virtual CString GetDialogName() const override;\n\n    DECLARE_MESSAGE_MAP()\npublic:\n    virtual BOOL OnInitDialog();\n    virtual BOOL PreTranslateMessage(MSG* pMsg);\n    afx_msg void OnTcnSelchangeTab1(NMHDR* pNMHDR, LRESULT* pResult);\n    afx_msg void OnSize(UINT nType, int cx, int cy);\n};\n"
  },
  {
    "path": "TrafficMonitor/HistoryTrafficFile.cpp",
    "content": "#include \"stdafx.h\"\n#include \"HistoryTrafficFile.h\"\n#include \"Common.h\"\n\nCHistoryTrafficFile::CHistoryTrafficFile(const wstring& file_path)\n\t: m_file_path(file_path)\n{\n}\n\nCHistoryTrafficFile::~CHistoryTrafficFile()\n{\n}\n\nvoid CHistoryTrafficFile::Save() const\n{\n\tofstream file{ m_file_path };\n\tchar buff[64];\n\tsprintf_s(buff, \"lines: \\\"%u\\\"\", static_cast<unsigned int>(m_history_traffics.size()));\t\t\t//ڵһд\n\tfile << buff << std::endl;\n\tfor (const auto& history_traffic : m_history_traffics)\n\t{\n\t\tif (history_traffic.mixed)\n\t\t\tsprintf_s(buff, \"%.4d/%.2d/%.2d %llu\", history_traffic.year, history_traffic.month, history_traffic.day, history_traffic.down_kBytes);\n\t\telse\n\t\t\tsprintf_s(buff, \"%.4d/%.2d/%.2d %llu/%llu\", history_traffic.year, history_traffic.month, history_traffic.day, history_traffic.up_kBytes, history_traffic.down_kBytes);\n\t\tfile << buff << std::endl;\n\t}\n\tfile.close();\n}\n\nvoid CHistoryTrafficFile::Load()\n{\n\tifstream file{ m_file_path };\n\tstring current_line, temp;\n\tHistoryTraffic traffic;\n\t//bool first_line{ true };\n\tif (CCommon::FileExist(m_file_path.c_str()))\n\t{\n\t\twhile (!file.eof())\n\t\t{\n\t\t\tif (m_history_traffics.size() > 9999) break;\t\t//ȡ10000ʷ¼\n\t\t\tstd::getline(file, current_line);\n\t\t\t//if (first_line)\n\t\t\t//{\n\t\t\t//\tfirst_line = false;\n\t\t\t//\tsize_t index = current_line.find(\"lines:\");\n\t\t\t//\tif(index != wstring::npos)\n\t\t\t//\t{\n\t\t\t//\t\tindex = current_line.find(\"\\\"\", index + 6);\n\t\t\t//\t\tsize_t index1 = current_line.find(\"\\\"\", index + 1);\n\t\t\t//\t\ttemp = current_line.substr(index + 1, index1 - index - 1);\n\t\t\t//\t\tm_size = atoll(temp.c_str());\n\t\t\t//\t\tcontinue;\n\t\t\t//\t}\n\t\t\t//}\n\n\t\t\tif (current_line.size() < 12) continue;\n\t\t\ttemp = current_line.substr(0, 4);\n\t\t\ttraffic.year = atoi(temp.c_str());\n\t\t\tif (traffic.year < 1900 || traffic.year > 3000)\n\t\t\t\tcontinue;\n\t\t\ttemp = current_line.substr(5, 2);\n\t\t\ttraffic.month = atoi(temp.c_str());\n\t\t\tif (traffic.month < 1 || traffic.month > 12)\n\t\t\t\tcontinue;\n\t\t\ttemp = current_line.substr(8, 2);\n\t\t\ttraffic.day = atoi(temp.c_str());\n\t\t\tif (traffic.day < 1 || traffic.day > 31)\n\t\t\t\tcontinue;\n\n\t\t\tint index = current_line.find(L'/', 11);\n\t\t\ttraffic.mixed = (index == wstring::npos);\n\t\t\tif (traffic.mixed)\n\t\t\t{\n\t\t\t\ttemp = current_line.substr(11);\n\t\t\t\ttraffic.down_kBytes = atoll(temp.c_str());\n\t\t\t\ttraffic.up_kBytes = 0;\n\t\t\t}\n\t\t\telse\n\t\t\t{\n\t\t\t\ttemp = current_line.substr(11, index - 11);\n\t\t\t\ttraffic.up_kBytes = atoll(temp.c_str());\n\t\t\t\ttemp = current_line.substr(index + 1);\n\t\t\t\ttraffic.down_kBytes = atoll(temp.c_str());\n\t\t\t}\n\t\t\tif (traffic.year > 0 && traffic.month > 0 && traffic.day > 0 && traffic.kBytes() > 0)\n\t\t\t\tm_history_traffics.push_back(traffic);\n\t\t}\n\t}\n\n\tMormalizeData();\n}\n\nvoid CHistoryTrafficFile::LoadSize()\n{\n\tifstream file{ m_file_path };\n\tstring current_line, temp;\n\tif (CCommon::FileExist(m_file_path.c_str()))\n\t{\n\t\tstd::getline(file, current_line);\t\t\t//ȡһ\n\t\tsize_t index = current_line.find(\"lines:\");\n\t\tif (index != wstring::npos)\n\t\t{\n\t\t\tindex = current_line.find(\"\\\"\", index + 6);\n\t\t\tsize_t index1 = current_line.find(\"\\\"\", index + 1);\n\t\t\ttemp = current_line.substr(index + 1, index1 - index - 1);\n\t\t\tm_size = atoll(temp.c_str());\n\t\t}\n\t}\n}\n\nvoid CHistoryTrafficFile::Merge(const CHistoryTrafficFile& history_traffic, bool ignore_same_data)\n{\n\tfor (const HistoryTraffic& traffic : history_traffic.m_history_traffics)\n\t{\n\t\tif(ignore_same_data)\n\t\t{\n\t\t\t//Ҫͬڵʹöַͬҵˣ\n\t\t\tif (std::binary_search(m_history_traffics.begin(), m_history_traffics.end(), traffic, HistoryTraffic::DateGreater))\n\t\t\t{\n\t\t\t\tauto iter = std::lower_bound(m_history_traffics.begin(), m_history_traffics.end(), traffic, HistoryTraffic::DateGreater);\n\t\t\t\tif (iter != m_history_traffics.end())\n\t\t\t\t{\n\t\t\t\t\tcontinue;\n\t\t\t\t}\n\t\t\t}\n\t\t}\n\t\tm_history_traffics.push_back(traffic);\n\t}\n\tMormalizeData();\n}\n\nvoid CHistoryTrafficFile::MormalizeData()\n{\n\tSYSTEMTIME current_time;\n\tGetLocalTime(&current_time);\n\tHistoryTraffic traffic;\n\ttraffic.year = current_time.wYear;\n\ttraffic.month = current_time.wMonth;\n\ttraffic.day = current_time.wDay;\n\ttraffic.up_kBytes = 0;\n\ttraffic.down_kBytes = 0;\n\ttraffic.mixed = false;\n\n\tif (m_history_traffics.empty())\n\t{\n\t\tm_history_traffics.push_front(traffic);\n\t}\n\n\tif (m_history_traffics.size() >= 2)\n\t{\n\t\t//ȡʷбڴӴС\n\t\tstd::sort(m_history_traffics.begin(), m_history_traffics.end(), HistoryTraffic::DateGreater);\n\n\t\t//бͬڵĿϲ\n\t\tfor (int i{}; i < static_cast<int>(m_history_traffics.size() - 1); i++)\n\t\t{\n\t\t\tif (HistoryTraffic::DateEqual(m_history_traffics[i], m_history_traffics[i + 1]))\n\t\t\t{\n\t\t\t\tm_history_traffics[i].up_kBytes += m_history_traffics[i + 1].up_kBytes;\n\t\t\t\tm_history_traffics[i].down_kBytes += m_history_traffics[i + 1].down_kBytes;\n\t\t\t\tm_history_traffics.erase(m_history_traffics.begin() + i + 1);\n\t\t\t}\n\t\t}\n\t}\n\n\t//бһĿǽ죬򽫵һĿͳƵΪʹõбǰһΪĿ\n\tif (HistoryTraffic::DateEqual(m_history_traffics[0], traffic))\n\t{\n\t\tm_today_up_traffic = static_cast<__int64>(m_history_traffics[0].up_kBytes) * 1024;\n\t\tm_today_down_traffic = static_cast<__int64>(m_history_traffics[0].down_kBytes) * 1024;\n\t\tm_history_traffics[0].mixed = false;\n\t}\n\telse\n\t{\n\t\tm_history_traffics.push_front(traffic);\n\t}\n\tm_size = m_history_traffics.size();\n}\n"
  },
  {
    "path": "TrafficMonitor/HistoryTrafficFile.h",
    "content": "#pragma once\n#include \"CommonData.h\"\nclass CHistoryTrafficFile\n{\npublic:\n\tCHistoryTrafficFile(const wstring& file_path);\n\t~CHistoryTrafficFile();\n\n\tvoid Save() const;\n\tvoid Load();\n\tvoid LoadSize();\t\t\t//ȡļĴС\n\tvoid Merge(const CHistoryTrafficFile& history_traffic, bool ignore_same_data = false);\t\t//ϲһCHistoryTrafficFileignore_same_dataΪtrueͬڵͬڵ\n\n\tconst wstring& GetFilePath() const { return m_file_path; }\n\tconst void SetFilePath(const wstring& file_path) { m_file_path = file_path; }\n\tdeque<HistoryTraffic>& GetTraffics() { return m_history_traffics; }\n\t__int64 GetTodayUpTraffic() const { return m_today_up_traffic; }\n\t__int64 GetTodayDownTraffic() const { return m_today_down_traffic; }\n\tsize_t Size() { return m_size; }\n\nprivate:\n\tvoid MormalizeData();\t\t//ʷ򲢺ϲͬ\n\nprivate:\n\twstring m_file_path;\n\tdeque<HistoryTraffic> m_history_traffics;\t//ʷ\n\t__int64 m_today_up_traffic{};\t//ʹõϴ\n\t__int64 m_today_down_traffic{};\t//ʹõ\n\tsize_t m_size{};\t\t\t\t//ݵ\n};\n\n"
  },
  {
    "path": "TrafficMonitor/HistoryTrafficListCtrl.cpp",
    "content": "﻿#include \"stdafx.h\"\n#include \"HistoryTrafficListCtrl.h\"\n\nIMPLEMENT_DYNAMIC(CHistoryTrafficListCtrl, CListCtrl)\n\nCHistoryTrafficListCtrl::CHistoryTrafficListCtrl()\n{\n}\n\n\nCHistoryTrafficListCtrl::~CHistoryTrafficListCtrl()\n{\n}\nvoid CHistoryTrafficListCtrl::SetDrawItemRangeData(int item, double range, COLORREF color)\n{\n\tif (item < 0) return;\n\tif (item >= static_cast<int>(m_item_rage_data.size()))\n\t\tm_item_rage_data.resize(item + 1);\n\tm_item_rage_data[item].data_value = range;\n\tm_item_rage_data[item].color = color;\n}\n\nvoid CHistoryTrafficListCtrl::SetDrawItemRangInLogScale(bool log_scale)\n{\n\tm_use_log_scale = log_scale;\n\tInvalidate();\n}\n\nBEGIN_MESSAGE_MAP(CHistoryTrafficListCtrl, CListCtrl)\n\tON_NOTIFY_REFLECT(NM_CUSTOMDRAW, &CHistoryTrafficListCtrl::OnNMCustomdraw)\nEND_MESSAGE_MAP()\n\n\nvoid CHistoryTrafficListCtrl::OnNMCustomdraw(NMHDR *pNMHDR, LRESULT *pResult)\n{\n\tif (m_draw_item_range)\n\t{\n\t\t*pResult = CDRF_DODEFAULT;\n\t\tLPNMLVCUSTOMDRAW lplvdr = reinterpret_cast<LPNMLVCUSTOMDRAW>(pNMHDR);\n\t\tNMCUSTOMDRAW& nmcd = lplvdr->nmcd;\n\t\tswitch (lplvdr->nmcd.dwDrawStage)\t//判断状态   \n\t\t{\n\t\tcase CDDS_PREPAINT:\n\t\t\t*pResult = CDRF_NOTIFYITEMDRAW;\n\t\t\tbreak;\n\t\tcase CDDS_ITEMPREPAINT:\t\t\t//如果为画ITEM之前就要进行颜色的改变\n\t\t\tif (nmcd.dwItemSpec >= 0 && nmcd.dwItemSpec < m_item_rage_data.size())\n\t\t\t{\n\t\t\t\tdouble range = m_item_rage_data[nmcd.dwItemSpec].data_value;\n\t\t\t\tCDC* pDC = CDC::FromHandle(nmcd.hdc);\t\t//获取绘图DC\n\t\t\t\tCRect item_rect, draw_rect;\n\t\t\t\tGetSubItemRect(nmcd.dwItemSpec,m_draw_item_range_row, LVIR_BOUNDS, item_rect);\t//获取绘图单元格的矩形区域\n\t\t\t\tCDrawCommon::SetDrawRect(pDC, item_rect);\t\t//设置绘图区域为当前列\n                //使用双缓冲绘图\n                {\n                    CDrawDoubleBuffer draw_double_buffer(pDC, item_rect);\n                    //填充背景\n                    draw_rect = item_rect;\n                    draw_rect.MoveToXY(0, 0);\n                    draw_double_buffer.GetMemDC()->FillSolidRect(draw_rect, GetSysColor(COLOR_WINDOW));\n                    if (draw_rect.Height() > 2 * m_margin)\n                    {\n                        draw_rect.top += m_margin;\n                        draw_rect.bottom -= m_margin;\n                    }\n                    int width;\n                    if (m_use_log_scale)\t//使用对数比例（y=ln(x+1)）\n                    {\n                        range = std::log(range + 1);\n                        width = static_cast<int>(range*draw_rect.Width() / std::log(1000 + 1));\n                    }\n                    else\t\t//使用线性比例（y=x）\n                    {\n                        width = static_cast<int>(range*draw_rect.Width() / 1000);\n                    }\n                    draw_rect.right = draw_rect.left + width;\n                    draw_double_buffer.GetMemDC()->FillSolidRect(draw_rect, m_item_rage_data[nmcd.dwItemSpec].color);\n                }\n\n\t\t\t\t//当前列绘制完成后将绘图区域设置为左边的区域，防止当前列的区域被覆盖\n\t\t\t\tCRect rect1{ item_rect };\n\t\t\t\trect1.left = 0;\n\t\t\t\trect1.right = item_rect.left;\n\t\t\t\tCDrawCommon::SetDrawRect(pDC, rect1);\n\t\t\t}\n\t\t\t*pResult = CDRF_DODEFAULT;\n\t\t\tbreak;\n\t\t}\n\t}\n}\n"
  },
  {
    "path": "TrafficMonitor/HistoryTrafficListCtrl.h",
    "content": "#pragma once\n#include \"afxcmn.h\"\n#include \"DrawCommon.h\"\nclass CHistoryTrafficListCtrl :\n\tpublic CListCtrl\n{\nDECLARE_DYNAMIC(CHistoryTrafficListCtrl)\npublic:\n\tCHistoryTrafficListCtrl();\n\t~CHistoryTrafficListCtrl();\n\n\tvoid EnableDrawItemRange(bool draw = true) { m_draw_item_range = draw; }\t//ǷҪĳһеԪлƱʾֵСľ\n\tvoid SetDrawItemRangeRow(int row) { m_draw_item_range_row = row; }\t\t\t//ҪƱʾֵСľεУעбΪұߵУ򣬴ұߵн޷ʾ\n\tvoid SetDrawItemRangeData(int item, double range, COLORREF color);\t\t\t//ĳһеĿݴСȡֵΪ1~1000ɫ\n\tvoid SetDrawItemRangMargin(int margin) { m_margin = margin; }\t\t\t//ûƻƵľεıԵԪ߿ľΣֵԽƵľԽϸǲܳбоһ\n\tvoid SetDrawItemRangInLogScale(bool log_scale);\t\t\t\t//ҪƱʾֵСľʱǷʹöʹԱ\n\nprotected:\n\tstruct ItemData\n\t{\n\t\tdouble data_value;\t\t//ҪƵľαʾֵСΧΪ0~1000\n\t\tCOLORREF color;\t\t//ҪƵľεɫ\n\t};\n\tbool m_draw_item_range{ false };\t\t//ǷҪĳһеԪлƱʾֵСľ\n\tint m_draw_item_range_row{};\t\t//ҪƱʾֵСľε\n\tint m_margin{};\n\tvector<ItemData> m_item_rage_data;\t//ڱʾÿһҪƵݴСͻͼɫ\n\tbool m_use_log_scale{ false };\t\t//ΪtrueʹöƾΣʹԱ\n\n\tDECLARE_MESSAGE_MAP()\n\tafx_msg void OnNMCustomdraw(NMHDR *pNMHDR, LRESULT *pResult);\n};\n\n"
  },
  {
    "path": "TrafficMonitor/HistoryTrafficListDlg.cpp",
    "content": "﻿// HistoryTrafficList.cpp: 实现文件\n//\n\n#include \"stdafx.h\"\n#include \"TrafficMonitor.h\"\n#include \"HistoryTrafficListDlg.h\"\n#include \"afxdialogex.h\"\n#include \"CalendarHelper.h\"\n\n\n// CHistoryTrafficListDlgDlg 对话框\n\nIMPLEMENT_DYNAMIC(CHistoryTrafficListDlg, CTabDlg)\n\nCHistoryTrafficListDlg::CHistoryTrafficListDlg(deque<HistoryTraffic>& history_traffics, CWnd* pParent /*=nullptr*/)\n\t: CTabDlg(IDD_HISTORY_TRAFFIC_LIST_DIALOG, pParent), m_history_traffics(history_traffics)\n{\n\n}\n\nCHistoryTrafficListDlg::~CHistoryTrafficListDlg()\n{\n}\n\n\nbool CHistoryTrafficListDlg::CalculateColumeWidth(std::vector<int>& widths)\n{\n    if (!IsWindow(m_history_list.GetSafeHwnd()))\n        return false;\n    CRect rect;\n    m_history_list.GetWindowRect(rect);\n    if (rect.Width() <= 0)\n        return false;\n    const int MAX_COLUME{ 5 };\n\tint width_date;\t\t//“日期”列的宽度\n    int width0;\t\t\t//“上传”、“下载”、“总流量”列的宽度\n    int width1;\t\t\t//“图表”列的宽度\n\twidth_date = rect.Width() * 3 / 11;\n    width0 = rect.Width() * 2 / 11;\n    if (width_date > theApp.DPI(150))\n\t\twidth_date = theApp.DPI(150);\n\tif (width0 > theApp.DPI(120))\n\t\twidth0 = theApp.DPI(120);\n    width1 = rect.Width() - (MAX_COLUME - 2) * width0 - width_date - theApp.DPI(20) - 1;\n    widths.resize(MAX_COLUME);\n\twidths[0] = width_date;\n    widths[1] = widths[2] = widths[3] = width0;\n    widths[4] = width1;\n    return true;\n}\n\nvoid CHistoryTrafficListDlg::AddListRow(const ListRowData& data, unsigned __int64 max_traffic)\n{\n    unsigned __int64 total_kbytes = data.up_kBytes + data.down_kBytes;\n    int index = m_history_list.GetItemCount();\n    m_history_list.InsertItem(index, data.str);\n    if (data.mixed)\n    {\n        m_history_list.SetItemText(index, 1, _T(\"-\"));\n        m_history_list.SetItemText(index, 2, _T(\"-\"));\n    }\n    else\n    {\n        m_history_list.SetItemText(index, 1, CCommon::KBytesToString(data.up_kBytes));\n        m_history_list.SetItemText(index, 2, CCommon::KBytesToString(data.down_kBytes));\n    }\n    m_history_list.SetItemText(index, 3, CCommon::KBytesToString(total_kbytes));\n\n    double range = static_cast<double>(total_kbytes) * 1000 / max_traffic;\n    COLORREF color;\n    if (total_kbytes < 1024 * 1024)\t\t//流量小于1GB时绘制蓝色\n        color = TRAFFIC_COLOR_BLUE;\n    else if (total_kbytes < 10 * 1024 * 1024)\t//流量小于10GB时绘制绿色\n        color = TRAFFIC_COLOR_GREEN;\n    else if (total_kbytes < 100 * 1024 * 1024)\t//流量小于100GB时绘制黄色\n        color = TRAFFIC_COLOR_YELLOE;\n    else if (total_kbytes < 1024 * 1024 * 1024)\t//流量小于1TB时绘制红色\n        color = TRAFFIC_COLOR_RED;\n    else\t\t//流量超过1TB时显示深红色\n        color = TRAFFIC_COLOR_DARK_RED;\n    m_history_list.SetDrawItemRangeData(index, range, color);\n}\n\nvoid CHistoryTrafficListDlg::ShowListData()\n{\n    m_history_list.DeleteAllItems();\n\n    //显示日视图\n    if(theApp.m_cfg_data.m_view_type == HistoryTrafficViewType::HV_DAY)\n    {\n        //获取历史流量列表中流量的最大值\n        unsigned __int64 max_traffic{};\n        for (const auto& traffic : m_history_traffics)\n        {\n            if (traffic.kBytes() > max_traffic)\n                max_traffic = traffic.kBytes();\n        }\n\n        for (size_t i{}; i < m_history_traffics.size(); i++)\n        {\n            CString date_str;\n            //CString k_bytes_str;\n            date_str.Format(_T(\"%.4d/%.2d/%.2d (\"), m_history_traffics[i].year, m_history_traffics[i].month, m_history_traffics[i].day);\n\n            int week_day = CCalendarHelper::CaculateWeekDay(m_history_traffics[i].year, m_history_traffics[i].month, m_history_traffics[i].day);\n            switch (week_day)\n            {\n            case 0:\n                date_str += CCommon::LoadText(IDS_SUNDAY);\n                break;\n            case 1:\n                date_str += CCommon::LoadText(IDS_MONDAY);\n                break;\n            case 2:\n                date_str += CCommon::LoadText(IDS_TUESDAY);\n                break;\n            case 3:\n                date_str += CCommon::LoadText(IDS_WEDNESDAY);\n                break;\n            case 4:\n                date_str += CCommon::LoadText(IDS_THURSDAY);\n                break;\n            case 5:\n                date_str += CCommon::LoadText(IDS_FRIDAY);\n                break;\n            case 6:\n                date_str += CCommon::LoadText(IDS_SATURDAY);\n                break;\n            default:\n                break;\n            }\n            date_str += _T(')');\n\n            ListRowData data;\n            data.str = date_str;\n            data.up_kBytes = m_history_traffics[i].up_kBytes;\n            data.down_kBytes = m_history_traffics[i].down_kBytes;\n            data.mixed = m_history_traffics[i].mixed;\n            AddListRow(data, max_traffic);\n        }\n    }\n\n    //显示月/季/年视图\n    else\n    {\n        std::vector<ListRowData> list_data;\n        unsigned __int64 max_traffic{};\n        for (const auto& traffic : m_history_traffics)\n        {\n            CString date_str;\n            if (theApp.m_cfg_data.m_view_type == HistoryTrafficViewType::HV_WEEK)\n            {\n                date_str.Format(_T(\"%.4d/\"), traffic.year);\n                date_str += CCommon::LoadTextFormat(IDS_WEEK_NUM, { traffic.week() });\n            }\n            else if (theApp.m_cfg_data.m_view_type == HistoryTrafficViewType::HV_MONTH)      //月视图\n            {\n                date_str.Format(_T(\"%.4d/%.2d\"), traffic.year, traffic.month);\n            }\n            else if (theApp.m_cfg_data.m_view_type == HistoryTrafficViewType::HV_QUARTER)   //季视图\n            {\n                date_str.Format(_T(\"%.4d/\"), traffic.year);\n                if (traffic.month <= 3)\n                    date_str += _T(\"Q1\");\n                else if (traffic.month <= 6)\n                    date_str += _T(\"Q2\");\n                else if (traffic.month <= 9)\n                    date_str += _T(\"Q3\");\n                else\n                    date_str += _T(\"Q4\");\n            }\n            else    //年视图\n            {\n                date_str.Format(_T(\"%.4d\"), traffic.year);\n            }\n            if (list_data.empty() || list_data.back().str != date_str)\n            {\n                if(!list_data.empty())\n                {\n                    unsigned __int64 cur_traffic{ list_data.back().up_kBytes + list_data.back().down_kBytes };\n                    if (cur_traffic > max_traffic)\n                        max_traffic = cur_traffic;\n                }\n                ListRowData data;\n                data.str = date_str;\n                data.up_kBytes = traffic.up_kBytes;\n                data.down_kBytes = traffic.down_kBytes;\n                list_data.push_back(data);\n            }\n            else\n            {\n                list_data.back().up_kBytes += traffic.up_kBytes;\n                list_data.back().down_kBytes += traffic.down_kBytes;\n            }\n        }\n\n        for (const auto& data : list_data)\n        {\n            AddListRow(data, max_traffic);\n        }\n    }\n}\n\nvoid CHistoryTrafficListDlg::DoDataExchange(CDataExchange* pDX)\n{\n    CTabDlg::DoDataExchange(pDX);\n    DDX_Control(pDX, IDC_HISTORY_INFO_LIST, m_history_list);\n    DDX_Control(pDX, IDC_VIEW_TYPE_COMBO, m_view_type_combo);\n    DDX_Control(pDX, IDC_VIEW_SCALE_COMBO, m_view_scale_combo);\n}\n\n\nBEGIN_MESSAGE_MAP(CHistoryTrafficListDlg, CTabDlg)\n\tON_WM_INITMENU()\n\tON_COMMAND(ID_USE_LINEAR_SCALE, &CHistoryTrafficListDlg::OnUseLinearScale)\n\tON_COMMAND(ID_USE_LOG_SCALE, &CHistoryTrafficListDlg::OnUseLogScale)\n    ON_WM_SIZE()\n    ON_CBN_SELCHANGE(IDC_VIEW_TYPE_COMBO, &CHistoryTrafficListDlg::OnCbnSelchangeViewTypeCombo)\n    ON_CBN_SELCHANGE(IDC_VIEW_SCALE_COMBO, &CHistoryTrafficListDlg::OnCbnSelchangeViewScaleCombo)\nEND_MESSAGE_MAP()\n\n\n// CHistoryTrafficListDlgDlg 消息处理程序\n\n\nBOOL CHistoryTrafficListDlg::OnInitDialog()\n{\n\tCTabDlg::OnInitDialog();\n\n\t// TODO:  在此添加额外的初始化\n\n\t//初始化列表控件\n\tm_history_list.SetExtendedStyle(LVS_EX_FULLROWSELECT | LVS_EX_LABELTIP);\n    std::vector<int> widths;\n    CalculateColumeWidth(widths);\n    m_history_list.InsertColumn(0, CCommon::LoadText(IDS_DATE), LVCFMT_LEFT, widths[0]);\n\tm_history_list.InsertColumn(1, CCommon::LoadText(IDS_UPLOAD), LVCFMT_RIGHT, widths[1]);\n\tm_history_list.InsertColumn(2, CCommon::LoadText(IDS_DOWNLOAD), LVCFMT_RIGHT, widths[2]);\n\tm_history_list.InsertColumn(3, CCommon::LoadText(IDS_TRAFFIC_USED), LVCFMT_RIGHT, widths[3]);\n\tm_history_list.InsertColumn(4, CCommon::LoadText(IDS_FIGURE), LVCFMT_LEFT, widths[4]);\n\n    m_history_list.EnableDrawItemRange();\n    m_history_list.SetDrawItemRangeRow(4);\n    m_history_list.SetDrawItemRangMargin(theApp.DPI(4));\n    m_history_list.SetDrawItemRangInLogScale(theApp.m_cfg_data.m_use_log_scale);\n\n    //初始化控件\n    m_view_type_combo.AddString(CCommon::LoadText(IDS_DAY_VIEW));\n    m_view_type_combo.AddString(CCommon::LoadText(IDS_WEEK_VIEW));\n    m_view_type_combo.AddString(CCommon::LoadText(IDS_MONTH_VIEW));\n    m_view_type_combo.AddString(CCommon::LoadText(IDS_QUARTER_VIEW));\n    m_view_type_combo.AddString(CCommon::LoadText(IDS_YEAR_VIEW));\n    m_view_type_combo.SetCurSel(static_cast<int>(theApp.m_cfg_data.m_view_type));\n\n    m_view_scale_combo.AddString(CCommon::LoadText(IDS_LINEAR_SCALE));\n    m_view_scale_combo.AddString(CCommon::LoadText(IDS_LOG_SCALE));\n    m_view_scale_combo.SetCurSel(theApp.m_cfg_data.m_use_log_scale ? 1 : 0);\n\n    //显示列表数据\n    ShowListData();\n\n\tm_menu.LoadMenu(IDR_HISTORY_TRAFFIC_MENU);\n\n\treturn TRUE;  // return TRUE unless you set the focus to a control\n\t\t\t\t  // 异常: OCX 属性页应返回 FALSE\n}\n\n\nBOOL CHistoryTrafficListDlg::OnNotify(WPARAM wParam, LPARAM lParam, LRESULT* pResult)\n{\n\t// TODO: 在此添加专用代码和/或调用基类\n\n\tLPNMHDR lpnmhdr = (LPNMHDR)lParam;\n\tif (lpnmhdr->code == NM_RCLICK)\n\t{\n\t\tCPoint point;\n\t\tGetCursorPos(&point);//获得光标的位置\n\t\tm_history_list.ScreenToClient(&point);//获得list控件在窗口上的坐标\n\t\tCWnd* pWnd = m_history_list.ChildWindowFromPoint(point);\n\t\tCHeaderCtrl* pHeader = m_history_list.GetHeaderCtrl();//获取列表视图控件的标题控件\n\t\tCRect item_rect;\n\t\tpHeader->GetItemRect(4, item_rect);\t\t//获取列表标题控件第2列的矩形区域，只有光标在第4列点击时才弹出右键菜单\n\t\tif (pWnd && (pWnd->GetSafeHwnd() == pHeader->GetSafeHwnd()) && item_rect.PtInRect(point))\n\t\t{\n\t\t\tHDHITTESTINFO info{};\n\t\t\tinfo.pt = point;\n\t\t\tpHeader->SendMessage(HDM_HITTEST, 0, (LPARAM)&info);\n\t\t\tCMenu * pMenu = m_menu.GetSubMenu(0);\n\t\t\tGetCursorPos(&point);\n\t\t\tpMenu->TrackPopupMenu(TPM_RIGHTBUTTON, point.x, point.y, this);\n\t\t}\n\t}\n\treturn CTabDlg::OnNotify(wParam, lParam, pResult);\n}\n\n\nvoid CHistoryTrafficListDlg::OnInitMenu(CMenu* pMenu)\n{\n\tCTabDlg::OnInitMenu(pMenu);\n\n\t// TODO: 在此处添加消息处理程序代码\n\tif (theApp.m_cfg_data.m_use_log_scale)\n\t\tpMenu->CheckMenuRadioItem(ID_USE_LINEAR_SCALE, ID_USE_LOG_SCALE, ID_USE_LOG_SCALE, MF_BYCOMMAND | MF_CHECKED);\n\telse\n\t\tpMenu->CheckMenuRadioItem(ID_USE_LINEAR_SCALE, ID_USE_LOG_SCALE, ID_USE_LINEAR_SCALE, MF_BYCOMMAND | MF_CHECKED);\n}\n\n\nvoid CHistoryTrafficListDlg::OnUseLinearScale()\n{\n\t// TODO: 在此添加命令处理程序代码\n\ttheApp.m_cfg_data.m_use_log_scale = false;\n\tm_history_list.SetDrawItemRangInLogScale(theApp.m_cfg_data.m_use_log_scale);\n}\n\n\nvoid CHistoryTrafficListDlg::OnUseLogScale()\n{\n\t// TODO: 在此添加命令处理程序代码\n\ttheApp.m_cfg_data.m_use_log_scale = true;\n\tm_history_list.SetDrawItemRangInLogScale(theApp.m_cfg_data.m_use_log_scale);\n}\n\n\nvoid CHistoryTrafficListDlg::OnSize(UINT nType, int cx, int cy)\n{\n    CTabDlg::OnSize(nType, cx, cy);\n\n    // TODO: 在此处添加消息处理程序代码\n    if (nType != SIZE_MINIMIZED)\n    {\n        std::vector<int> widths;\n        if(CalculateColumeWidth(widths))\n        {\n            for (size_t i{}; i < widths.size(); i++)\n            {\n                m_history_list.SetColumnWidth(i, widths[i]);\n            }\n        }\n    }\n}\n\n\nvoid CHistoryTrafficListDlg::OnCbnSelchangeViewTypeCombo()\n{\n    // TODO: 在此添加控件通知处理程序代码\n    theApp.m_cfg_data.m_view_type = static_cast<HistoryTrafficViewType>(m_view_type_combo.GetCurSel());\n    ShowListData();\n}\n\n\nvoid CHistoryTrafficListDlg::OnCbnSelchangeViewScaleCombo()\n{\n    // TODO: 在此添加控件通知处理程序代码\n    theApp.m_cfg_data.m_use_log_scale = (m_view_scale_combo.GetCurSel() != 0);\n    m_history_list.SetDrawItemRangInLogScale(theApp.m_cfg_data.m_use_log_scale);\n}\n"
  },
  {
    "path": "TrafficMonitor/IconSelectDlg.cpp",
    "content": "﻿// IconSelectDlg.cpp : 实现文件\n//\n\n#include \"stdafx.h\"\n#include \"TrafficMonitor.h\"\n#include \"IconSelectDlg.h\"\n#include \"afxdialogex.h\"\n\n\n// CIconSelectDlg 对话框\n\nIMPLEMENT_DYNAMIC(CIconSelectDlg, CBaseDialog)\n\nCIconSelectDlg::CIconSelectDlg(int icon_selected, CWnd* pParent /*=NULL*/)\n    : CBaseDialog(IDD_ICON_SELECT_DIALOG, pParent), m_icon_selected{ icon_selected }\n{\n\n}\n\nCIconSelectDlg::~CIconSelectDlg()\n{\n}\n\nint CIconSelectDlg::GetIconSelected() const\n{\n    if (m_icon_selected < 0 || m_icon_selected >= MAX_NOTIFY_ICON)\n        return 0;\n    return m_icon_selected;\n}\n\nvoid CIconSelectDlg::SetAutoAdaptNotifyIcon(bool val)\n{\n    m_atuo_adapt_notify_icon = val;\n}\n\nbool CIconSelectDlg::AutoAdaptNotifyIcon() const\n{\n    return m_atuo_adapt_notify_icon;\n}\n\nvoid CIconSelectDlg::DoDataExchange(CDataExchange* pDX)\n{\n    CBaseDialog::DoDataExchange(pDX);\n    DDX_Control(pDX, IDC_ICON_PREVIEW, m_preview_pic);\n    DDX_Control(pDX, IDC_COMBO1, m_icon_select_combo);\n    DDX_Control(pDX, IDC_AUTO_ADAPT_CHECK, m_auto_adapt_chk);\n}\n\nvoid CIconSelectDlg::DrawPreviewIcon(CDC* pDC)\n{\n    //pDC->FillSolidRect(CRect(CPoint(ICON_X, ICON_Y), CSize(theApp.DPI(16), theApp.DPI(16))), RGB(0, 0, 0));\n    //pDC->DrawIcon(ICON_X, ICON_Y, m_icons[m_icon_selected]);\n    ::DrawIconEx(pDC->m_hDC, ICON_X, ICON_Y, theApp.m_notify_icons[GetIconSelected()], theApp.DPI(16), theApp.DPI(16), 0, NULL, DI_NORMAL);\n}\n\n\nBEGIN_MESSAGE_MAP(CIconSelectDlg, CBaseDialog)\n    //ON_WM_TIMER()\n    ON_CBN_SELCHANGE(IDC_COMBO1, &CIconSelectDlg::OnCbnSelchangeCombo1)\n    ON_MESSAGE(WM_CONTROL_REPAINT, &CIconSelectDlg::OnControlRepaint)\n    ON_BN_CLICKED(IDC_AUTO_ADAPT_CHECK, &CIconSelectDlg::OnBnClickedAutoAdaptCheck)\nEND_MESSAGE_MAP()\n\n\n// CIconSelectDlg 消息处理程序\n\n\nBOOL CIconSelectDlg::OnInitDialog()\n{\n    CBaseDialog::OnInitDialog();\n\n    // TODO:  在此添加额外的初始化\n    SetWindowText(CCommon::LoadText(IDS_TITLE_CHANGE_ICON));\n\n    //设置预览图大小\n    m_preview_pic.SetWindowPos(nullptr, 0, 0, PREVIEW_WIDTH, PREVIEW_HEIGHT, SWP_NOZORDER | SWP_NOMOVE);\n    if (m_icon_selected == 4 || m_icon_selected == 5)\n        m_preview_pic.SetPicture((HBITMAP)LoadImage(AfxGetInstanceHandle(),\n            MAKEINTRESOURCE(IDB_NOTIFY_ICON_PREVIEW_LIGHT), IMAGE_BITMAP, 0, 0, LR_DEFAULTCOLOR | LR_CREATEDIBSECTION));\n    else\n        m_preview_pic.SetPicture((HBITMAP)LoadImage(AfxGetInstanceHandle(),\n            MAKEINTRESOURCE(IDB_NOTIFY_ICON_PREVIEW), IMAGE_BITMAP, 0, 0, LR_DEFAULTCOLOR | LR_CREATEDIBSECTION));\n\n    //初始化下拉列表\n    m_icon_select_combo.AddString(CCommon::LoadText(IDS_DEFAULT_ICON));\n    m_icon_select_combo.AddString(CCommon::LoadText(IDS_ICON, _T(\" 1\")));\n    m_icon_select_combo.AddString(CCommon::LoadText(IDS_ICON, _T(\" 2\")));\n    m_icon_select_combo.AddString(CCommon::LoadText(IDS_ICON, _T(\" 3\")));\n    m_icon_select_combo.AddString(CCommon::LoadText(IDS_ICON, _T(\" 4\")));\n    m_icon_select_combo.AddString(CCommon::LoadText(IDS_ICON, _T(\" 5\")));\n    m_icon_select_combo.SetCurSel(m_icon_selected);\n\n    m_auto_adapt_chk.SetCheck(m_atuo_adapt_notify_icon);\n    m_auto_adapt_chk.EnableWindow(theApp.m_win_version.GetMajorVersion() >= 10);\n\n    return TRUE;  // return TRUE unless you set the focus to a control\n                  // 异常: OCX 属性页应返回 FALSE\n}\n\n\nvoid CIconSelectDlg::OnCbnSelchangeCombo1()\n{\n    // TODO: 在此添加控件通知处理程序代码\n    m_icon_selected = m_icon_select_combo.GetCurSel();\n    if (m_icon_selected == 4 || m_icon_selected == 5)\n        m_preview_pic.SetPicture((HBITMAP)LoadImage(AfxGetInstanceHandle(),\n            MAKEINTRESOURCE(IDB_NOTIFY_ICON_PREVIEW_LIGHT), IMAGE_BITMAP, 0, 0, LR_DEFAULTCOLOR | LR_CREATEDIBSECTION));\n    else\n        m_preview_pic.SetPicture((HBITMAP)LoadImage(AfxGetInstanceHandle(),\n            MAKEINTRESOURCE(IDB_NOTIFY_ICON_PREVIEW), IMAGE_BITMAP, 0, 0, LR_DEFAULTCOLOR | LR_CREATEDIBSECTION));\n    DrawPreviewIcon(m_preview_pic.GetDC());\n}\n\n\nafx_msg LRESULT CIconSelectDlg::OnControlRepaint(WPARAM wParam, LPARAM lParam)\n{\n    CWnd* pControl = (CWnd*)wParam;\n    CDC* pDC = (CDC*)lParam;\n    if (pControl == &m_preview_pic)\n    {\n        //当收到m_preview_pic控件的重绘消息时，同时重绘图标\n        DrawPreviewIcon(pDC);\n    }\n    return 0;\n}\n\n\nvoid CIconSelectDlg::OnBnClickedAutoAdaptCheck()\n{\n    // TODO: 在此添加控件通知处理程序代码\n    m_atuo_adapt_notify_icon = (m_auto_adapt_chk.GetCheck() != 0);\n}\n\nCString CIconSelectDlg::GetDialogName() const\n{\n    return _T(\"\");\n}\n"
  },
  {
    "path": "TrafficMonitor/IconSelectDlg.h",
    "content": "﻿#pragma once\n#include \"afxwin.h\"\n#include \"PictureStatic.h\"\n#include \"BaseDialog.h\"\n\n// CIconSelectDlg 对话框\n\nclass CIconSelectDlg : public CBaseDialog\n{\n    DECLARE_DYNAMIC(CIconSelectDlg)\n\npublic:\n    CIconSelectDlg(int icon_selected, CWnd* pParent = NULL);   // 标准构造函数\n    virtual ~CIconSelectDlg();\n\n    int GetIconSelected() const;\n\n    void SetAutoAdaptNotifyIcon(bool val);\n    bool AutoAdaptNotifyIcon() const;\n\n    // 对话框数据\n#ifdef AFX_DESIGN_TIME\n    enum { IDD = IDD_ICON_SELECT_DIALOG };\n#endif\n\n#define PREVIEW_WIDTH theApp.DPI(200)\t\t//预览图的宽高\n#define PREVIEW_HEIGHT theApp.DPI(40)\n\n#define ICON_X theApp.DPI(46)\t\t//预览图中图标的位置\n#define ICON_Y theApp.DPI(12)\n\nprotected:\n    CPictureStatic m_preview_pic;\n    CComboBox m_icon_select_combo;\n    CButton m_auto_adapt_chk;\n\n    int m_icon_selected{};\n\n    bool m_atuo_adapt_notify_icon;\n\nprotected:\n    virtual void DoDataExchange(CDataExchange* pDX);    // DDX/DDV 支持\n    virtual CString GetDialogName() const override;\n\n    void DrawPreviewIcon(CDC* pDC);\n\n    DECLARE_MESSAGE_MAP()\npublic:\n    virtual BOOL OnInitDialog();\n    //afx_msg void OnTimer(UINT_PTR nIDEvent);\n    afx_msg void OnCbnSelchangeCombo1();\nprotected:\n    afx_msg LRESULT OnControlRepaint(WPARAM wParam, LPARAM lParam);\npublic:\n    afx_msg void OnBnClickedAutoAdaptCheck();\n};\n"
  },
  {
    "path": "TrafficMonitor/IniHelper.cpp",
    "content": "﻿#include \"stdafx.h\"\n#include \"IniHelper.h\"\n#include \"TrafficMonitor.h\"\n\n\nCIniHelper::CIniHelper(const wstring& file_path)\n{\n    m_file_path = file_path;\n    ifstream file_stream{ file_path };\n    if (file_stream.fail())\n    {\n        return;\n    }\n    //读取文件内容\n    string ini_str;\n    while (!file_stream.eof())\n    {\n        ini_str.push_back(file_stream.get());\n    }\n    ini_str.pop_back();\n    if (!ini_str.empty() && ini_str.back() != L'\\n')        //确保文件末尾有回车符\n        ini_str.push_back(L'\\n');\n    //判断文件是否是utf8编码\n    bool is_utf8;\n    if (ini_str.size() >= 3 && ini_str[0] == -17 && ini_str[1] == -69 && ini_str[2] == -65)\n    {\n        //如果有UTF8的BOM，则删除BOM\n        is_utf8 = true;\n        ini_str = ini_str.substr(3);\n    }\n    else\n    {\n        is_utf8 = false;\n    }\n    //转换成Unicode\n    m_ini_str = CCommon::StrToUnicode(ini_str.c_str(), is_utf8);\n}\n\n\nCIniHelper::~CIniHelper()\n{\n}\n\nvoid CIniHelper::SetSaveAsUTF8(bool utf8)\n{\n    m_save_as_utf8 = utf8;\n}\n\nvoid CIniHelper::WriteString(const wchar_t * AppName, const wchar_t * KeyName, const wstring& str)\n{\n    wstring write_str{ str };\n    if (!write_str.empty() && (write_str[0] == L' ' || write_str.back() == L' '))       //如果字符串前后含有空格，则在字符串前后添加引号\n    {\n        write_str = DEF_CH + write_str;\n        write_str.push_back(DEF_CH);\n    }\n    _WriteString(AppName, KeyName, write_str);\n}\n\nwstring CIniHelper::GetString(const wchar_t * AppName, const wchar_t * KeyName, const wchar_t* default_str) const\n{\n    wstring rtn{_GetString(AppName, KeyName, default_str)};\n    //如果读取的字符串前后有指定的字符，则删除它\n    if (!rtn.empty() && (rtn.front() == L'$' || rtn.front() == DEF_CH))\n        rtn = rtn.substr(1);\n    if (!rtn.empty() && (rtn.back() == L'$' || rtn.back() == DEF_CH))\n        rtn.pop_back();\n    return rtn;\n}\n\nvoid CIniHelper::WriteInt(const wchar_t * AppName, const wchar_t * KeyName, int value)\n{\n    wchar_t buff[16]{};\n    _itow_s(value, buff, 10);\n    _WriteString(AppName, KeyName, wstring(buff));\n}\n\nint CIniHelper::GetInt(const wchar_t * AppName, const wchar_t * KeyName, int default_value) const\n{\n    wchar_t default_str_buff[16]{};\n    _itow_s(default_value, default_str_buff, 10);\n    wstring rtn{ _GetString(AppName, KeyName, default_str_buff) };\n    return _ttoi(rtn.c_str());\n}\n\nvoid CIniHelper::WriteBool(const wchar_t * AppName, const wchar_t * KeyName, bool value)\n{\n    if(value)\n        _WriteString(AppName, KeyName, wstring(L\"true\"));\n    else\n        _WriteString(AppName, KeyName, wstring(L\"false\"));\n}\n\nbool CIniHelper::GetBool(const wchar_t * AppName, const wchar_t * KeyName, bool default_value) const\n{\n    wstring rtn{ _GetString(AppName, KeyName, (default_value ? L\"true\" : L\"false\")) };\n    if (rtn == L\"true\")\n        return true;\n    else if (rtn == L\"false\")\n        return false;\n    else\n        return (_ttoi(rtn.c_str()) != 0);\n}\n\nvoid CIniHelper::WriteIntArray(const wchar_t * AppName, const wchar_t * KeyName, const int * values, int size)\n{\n    CString str, tmp;\n    for (int i{}; i < size; i++)\n    {\n        tmp.Format(_T(\"%d,\"), values[i]);\n        str += tmp;\n    }\n    _WriteString(AppName, KeyName, wstring(str));\n}\n\nvoid CIniHelper::GetIntArray(const wchar_t * AppName, const wchar_t * KeyName, int * values, int size, int default_value) const\n{\n    CString default_str;\n    default_str.Format(_T(\"%d\"), default_value);\n    wstring str;\n    str = _GetString(AppName, KeyName, default_str);\n    std::vector<wstring> split_result;\n    CCommon::StringSplit(str, L',', split_result);\n    for (int i = 0; i < size; i++)\n    {\n        if (i < split_result.size())\n            values[i] = _wtoi(split_result[i].c_str());\n        else if (i > 0)\n            values[i] = values[i - 1];\n        else\n            values[i] = default_value;\n    }\n}\n\nvoid CIniHelper::WriteBoolArray(const wchar_t * AppName, const wchar_t * KeyName, const bool * values, int size)\n{\n    int value{};\n    for (int i{}; i < size; i++)\n    {\n        if (values[i])\n            value |= (1 << i);\n    }\n    return WriteInt(AppName, KeyName, value);\n}\n\nvoid CIniHelper::GetBoolArray(const wchar_t * AppName, const wchar_t * KeyName, bool * values, int size, bool default_value) const\n{\n    int value = GetInt(AppName, KeyName, 0);\n    for (int i{}; i < size; i++)\n    {\n        values[i] = ((value >> i) % 2 != 0);\n    }\n}\n\nvoid CIniHelper::WriteStringList(const wchar_t* AppName, const wchar_t* KeyName, const vector<wstring>& values)\n{\n    wstring str_write = MergeStringList(values);\n    _WriteString(AppName, KeyName, str_write);\n}\n\nvoid CIniHelper::GetStringList(const wchar_t* AppName, const wchar_t* KeyName, vector<wstring>& values, const vector<wstring>& default_value) const\n{\n    wstring default_str = MergeStringList(default_value);\n    wstring str_value = _GetString(AppName, KeyName, default_str.c_str());\n    SplitStringList(values, str_value);\n}\n\nvoid CIniHelper::SaveFontData(const wchar_t * AppName, const FontInfo & font)\n{\n    WriteString(AppName, L\"font_name\", wstring(font.name));\n    WriteInt(AppName, L\"font_size\", font.size);\n    bool style[4];\n    style[0] = font.bold;\n    style[1] = font.italic;\n    style[2] = font.underline;\n    style[3] = font.strike_out;\n    WriteBoolArray(AppName, L\"font_style\", style, 4);\n}\n\nbool CIniHelper::Save()\n{\n    ofstream file_stream{ m_file_path };\n    if(file_stream.fail())\n        return false;\n    string ini_str{ CCommon::UnicodeToStr(m_ini_str.c_str(), m_save_as_utf8) };\n    if (m_save_as_utf8)     //如果以UTF8编码保存，先插入BOM\n    {\n        string utf8_bom;\n        utf8_bom.push_back(-17);\n        utf8_bom.push_back(-69);\n        utf8_bom.push_back(-65);\n        file_stream << utf8_bom;\n    }\n\n    file_stream << ini_str;\n    return true;\n}\n\nvoid CIniHelper::LoadFontData(const wchar_t * AppName, FontInfo & font, const FontInfo& default_font) const\n{\n    font.name = GetString(AppName, L\"font_name\", default_font.name).c_str();\n    font.size = GetInt(AppName, L\"font_size\", default_font.size);\n    bool style[4];\n    GetBoolArray(AppName, L\"font_style\", style, 4);\n    font.bold = style[0];\n    font.italic = style[1];\n    font.underline = style[2];\n    font.strike_out = style[3];\n}\n\nvoid CIniHelper::LoadMainWndColors(const wchar_t * AppName, const wchar_t * KeyName, std::map<CommonDisplayItem, COLORREF>& text_colors, COLORREF default_color)\n{\n    CString default_str;\n    default_str.Format(_T(\"%d\"), default_color);\n    wstring str;\n    str = _GetString(AppName, KeyName, default_str);\n    std::vector<wstring> split_result;\n    CCommon::StringSplit(str, L',', split_result);\n    size_t index = 0;\n    for (auto iter = theApp.m_plugins.AllDisplayItemsWithPlugins().begin(); iter != theApp.m_plugins.AllDisplayItemsWithPlugins().end(); ++iter)\n    {\n        if (index < split_result.size())\n            text_colors[*iter] = _wtoi(split_result[index].c_str());\n        else if (!split_result.empty())\n            text_colors[*iter] = _wtoi(split_result[0].c_str());\n        else\n            text_colors[*iter] = default_color;\n        index++;\n    }\n}\n\nvoid CIniHelper::SaveMainWndColors(const wchar_t * AppName, const wchar_t * KeyName, const std::map<CommonDisplayItem, COLORREF>& text_colors)\n{\n    CString str;\n    for (auto iter = text_colors.begin(); iter != text_colors.end(); ++iter)\n    {\n        CString tmp;\n        tmp.Format(_T(\"%d,\"), iter->second);\n        str += tmp;\n    }\n    _WriteString(AppName, KeyName, wstring(str));\n\n}\n\nvoid CIniHelper::LoadTaskbarWndColors(const wchar_t * AppName, const wchar_t * KeyName, std::map<CommonDisplayItem, TaskbarItemColor>& text_colors, COLORREF default_color)\n{\n    CString default_str;\n    default_str.Format(_T(\"%d\"), default_color);\n    wstring str;\n    str = _GetString(AppName, KeyName, default_str);\n    std::vector<wstring> split_result;\n    CCommon::StringSplit(str, L',', split_result);\n    size_t index = 0;\n    for (auto iter = theApp.m_plugins.AllDisplayItemsWithPlugins().begin(); iter != theApp.m_plugins.AllDisplayItemsWithPlugins().end(); ++iter)\n    {\n        if (index < split_result.size())\n            text_colors[*iter].label = _wtoi(split_result[index].c_str());\n        else if (!split_result.empty())\n            text_colors[*iter].label = _wtoi(split_result[0].c_str());\n        else\n            text_colors[*iter].label = default_color;\n\n        if (index + 1 < split_result.size())\n            text_colors[*iter].value = _wtoi(split_result[index + 1].c_str());\n        else if (split_result.size() > 1)\n            text_colors[*iter].value = _wtoi(split_result[1].c_str());\n        else\n            text_colors[*iter].value = default_color;\n        index += 2;\n    }\n\n}\n\nvoid CIniHelper::SaveTaskbarWndColors(const wchar_t * AppName, const wchar_t * KeyName, const std::map<CommonDisplayItem, TaskbarItemColor>& text_colors)\n{\n    CString str;\n    for (auto iter = text_colors.begin(); iter != text_colors.end(); ++iter)\n    {\n        CString tmp;\n        tmp.Format(_T(\"%d,%d,\"), iter->second.label, iter->second.value);\n        str += tmp;\n    }\n    _WriteString(AppName, KeyName, wstring(str));\n}\n\nvoid CIniHelper::LoadPluginDisplayStr(bool is_main_window)\n{\n    DispStrings& disp_str{ is_main_window ? theApp.m_main_wnd_data.disp_str : theApp.m_taskbar_data.disp_str };\n    std::wstring app_name{ is_main_window ? L\"plugin_display_str_main_window\" : L\"plugin_display_str_taskbar_window\" };\n    for (const auto& plugin : theApp.m_plugins.GetPluginItems())\n    {\n        disp_str.Load(plugin->GetItemId(), GetString(app_name.c_str(), plugin->GetItemId(), plugin->GetItemLableText()));\n    }\n}\n\nvoid CIniHelper::SavePluginDisplayStr(bool is_main_window)\n{\n    DispStrings& disp_str{ is_main_window ? theApp.m_main_wnd_data.disp_str : theApp.m_taskbar_data.disp_str };\n    std::wstring app_name{ is_main_window ? L\"plugin_display_str_main_window\" : L\"plugin_display_str_taskbar_window\" };\n    for (const auto& plugin : theApp.m_plugins.GetPluginItems())\n    {\n        WriteString(app_name.c_str(), plugin->GetItemId(), disp_str.Get(plugin));\n    }\n}\n\nvoid CIniHelper::_WriteString(const wchar_t * AppName, const wchar_t * KeyName, const wstring & str)\n{\n    wstring app_str{ L\"[\" };\n    app_str.append(AppName).append(L\"]\");\n    size_t app_pos{}, app_end_pos, key_pos;\n    app_pos = m_ini_str.find(app_str);\n    if (app_pos == wstring::npos)       //找不到AppName，则在最后面添加\n    {\n        if (!m_ini_str.empty() && m_ini_str.back() != L'\\n')\n            m_ini_str += L\"\\n\";\n        app_pos = m_ini_str.size();\n        m_ini_str += app_str;\n        m_ini_str += L\"\\n\";\n    }\n    app_end_pos = m_ini_str.find(L\"\\n[\", app_pos + 2);\n    if (app_end_pos != wstring::npos)\n        app_end_pos++;\n\n    key_pos = m_ini_str.find(wstring(L\"\\n\") + KeyName + L' ', app_pos);     //查找“\\nkey_name ”\n    if (key_pos >= app_end_pos)     //如果找不到“\\nkey_name ”，则查找“\\nkey_name=”\n        key_pos = m_ini_str.find(wstring(L\"\\n\") + KeyName + L'=', app_pos);\n    if (key_pos >= app_end_pos)             //找不到KeyName，则插入一个\n    {\n        //wchar_t buff[256];\n        //swprintf_s(buff, L\"%s = %s\\n\", KeyName, str.c_str());\n        std::wstring str_temp = KeyName;\n        str_temp += L\" = \";\n        str_temp += str;\n        str_temp += L\"\\n\";\n        if (app_end_pos == wstring::npos)\n            m_ini_str += str_temp;\n        else\n            m_ini_str.insert(app_end_pos, str_temp);\n    }\n    else    //找到了KeyName，将等号到换行符之间的文本替换\n    {\n        size_t str_pos;\n        str_pos = m_ini_str.find(L'=', key_pos + 2);\n        size_t line_end_pos = m_ini_str.find(L'\\n', key_pos + 2);\n        if (str_pos > line_end_pos) //所在行没有等号，则插入一个等号\n        {\n            m_ini_str.insert(key_pos + wcslen(KeyName) + 1, L\" =\");\n            str_pos = key_pos + wcslen(KeyName) + 2;\n        }\n        else\n        {\n            str_pos++;\n        }\n        size_t str_end_pos;\n        str_end_pos = m_ini_str.find(L\"\\n\", str_pos);\n        m_ini_str.replace(str_pos, str_end_pos - str_pos, L\" \" + str);\n    }\n}\n\nwstring CIniHelper::_GetString(const wchar_t * AppName, const wchar_t * KeyName, const wchar_t* default_str) const\n{\n    wstring app_str{ L\"[\" };\n    app_str.append(AppName).append(L\"]\");\n    size_t app_pos{}, app_end_pos, key_pos;\n    app_pos = m_ini_str.find(app_str);\n    if (app_pos == wstring::npos)       //找不到AppName，返回默认字符串\n        return default_str;\n\n    app_end_pos = m_ini_str.find(L\"\\n[\", app_pos + 2);\n    if (app_end_pos != wstring::npos)\n        app_end_pos++;\n\n    key_pos = m_ini_str.find(wstring(L\"\\n\") + KeyName + L' ', app_pos);     //查找“\\nkey_name ”\n    if (key_pos >= app_end_pos)     //如果找不到“\\nkey_name ”，则查找“\\nkey_name=”\n        key_pos = m_ini_str.find(wstring(L\"\\n\") + KeyName + L'=', app_pos);\n    if (key_pos >= app_end_pos)             //找不到KeyName，返回默认字符串\n    {\n        return default_str;\n    }\n    else    //找到了KeyName，获取等号到换行符之间的文本\n    {\n        size_t str_pos;\n        str_pos = m_ini_str.find(L'=', key_pos + 2);\n        size_t line_end_pos = m_ini_str.find(L'\\n', key_pos + 2);\n        if (str_pos > line_end_pos) //所在行没有等号，返回默认字符串\n        {\n            return default_str;\n        }\n        else\n        {\n            str_pos++;\n        }\n        size_t str_end_pos;\n        str_end_pos = m_ini_str.find(L\"\\n\", str_pos);\n        //获取文本\n        wstring return_str{ m_ini_str.substr(str_pos, str_end_pos - str_pos) };\n        //如果前后有空格，则将其删除\n        CCommon::StringNormalize(return_str);\n        return return_str;\n    }\n}\n\nwstring CIniHelper::MergeStringList(const vector<wstring>& values)\n{\n    wstring str_merge;\n    int index = 0;\n    //在每个字符串前后加上引号，再将它们用逗号连接起来\n    for (const wstring& str : values)\n    {\n        if (index > 0)\n            str_merge.push_back(L',');\n        str_merge.push_back(L'\\\"');\n        str_merge += str;\n        str_merge.push_back(L'\\\"');\n        index++;\n    }\n    return str_merge;\n}\n\nvoid CIniHelper::SplitStringList(vector<wstring>& values, wstring str_value)\n{\n    CCommon::StringSplit(str_value, wstring(L\"\\\",\\\"\"), values);\n    if (!values.empty())\n    {\n        //结果中第一项前面和最后一项的后面各还有一个引号，将它们删除\n        values.front() = values.front().substr(1);\n        values.back().pop_back();\n    }\n}\n"
  },
  {
    "path": "TrafficMonitor/IniHelper.h",
    "content": "﻿//ini读写类\n//使用时将ini文件路径通过构造函数参数传递\n//在向ini文件写入数据时，需要在最后调用Save()函数以将更改保存到文件\n//默认以UTF8_BOM编码保存，如果要以ANSI保存，请调用SetSaveAsUTF8(false);\n#pragma once\n#include \"CommonData.h\"\n#include \"Common.h\"\n\nclass CIniHelper\n{\npublic:\n\tCIniHelper(const wstring& file_path);\n\t~CIniHelper();\n\n\tvoid SetSaveAsUTF8(bool utf8);\n\n\tvoid WriteString(const wchar_t* AppName, const wchar_t* KeyName, const wstring& str);\n\twstring GetString(const wchar_t* AppName, const wchar_t* KeyName, const wchar_t* default_str) const;\n\tvoid WriteInt(const wchar_t * AppName, const wchar_t * KeyName, int value);\n\tint GetInt(const wchar_t * AppName, const wchar_t * KeyName, int default_value) const;\n\tvoid WriteBool(const wchar_t * AppName, const wchar_t * KeyName, bool value);\n\tbool GetBool(const wchar_t * AppName, const wchar_t * KeyName, bool default_value) const;\n\tvoid WriteIntArray(const wchar_t * AppName, const wchar_t * KeyName, const int* values, int size);\t\t//写入一个int数组，元素个数为size\n\tvoid GetIntArray(const wchar_t * AppName, const wchar_t * KeyName, int* values, int size, int default_value = 0) const;\t\t//读取一个int数组，储存到values，元素个数为size\n\tvoid WriteBoolArray(const wchar_t * AppName, const wchar_t * KeyName, const bool* values, int size);\n\tvoid GetBoolArray(const wchar_t * AppName, const wchar_t * KeyName, bool* values, int size, bool default_value = false) const;\n    void WriteStringList(const wchar_t* AppName, const wchar_t* KeyName, const vector<wstring>& values);      //写入一个字符串列表，由于保存到ini文件中时字符串前后会加上引号，所以字符串中不能包含引号\n    void GetStringList(const wchar_t* AppName, const wchar_t* KeyName, vector<wstring>& values, const vector<wstring>& default_value) const;\n\n\tvoid SaveFontData(const wchar_t * AppName, const FontInfo& font);\n\tvoid LoadFontData(const wchar_t * AppName, FontInfo& font, const FontInfo& default_font) const;\n\n    void LoadMainWndColors(const wchar_t * AppName, const wchar_t * KeyName, std::map<CommonDisplayItem, COLORREF>& text_colors, COLORREF default_color);\n    void SaveMainWndColors(const wchar_t * AppName, const wchar_t * KeyName, const std::map<CommonDisplayItem, COLORREF>& text_colors);\n\n    void LoadTaskbarWndColors(const wchar_t * AppName, const wchar_t * KeyName, std::map<CommonDisplayItem, TaskbarItemColor>& text_colors, COLORREF default_color);\n    void SaveTaskbarWndColors(const wchar_t * AppName, const wchar_t * KeyName, const std::map<CommonDisplayItem, TaskbarItemColor>& text_colors);\n\n    void LoadPluginDisplayStr(bool is_main_window);\n    void SavePluginDisplayStr(bool is_main_window);\n\n\tbool Save();\t\t//将ini文件保存到文件，成功返回true\n\nprotected:\n\twstring m_file_path;\n\twstring m_ini_str;\n\tbool m_save_as_utf8{ true };\t\t//是否以及UTF8编码保存\n\n\tvoid _WriteString(const wchar_t* AppName, const wchar_t* KeyName, const wstring& str);\n\twstring _GetString(const wchar_t* AppName, const wchar_t* KeyName, const wchar_t* default_str) const;\n\n    static wstring MergeStringList(const vector<wstring>& values);\n    static void SplitStringList(vector<wstring>& values, wstring str_value);\n};\n"
  },
  {
    "path": "TrafficMonitor/ListCtrlEx.cpp",
    "content": "#include \"stdafx.h\"\n#include \"ListCtrlEx.h\"\n\nIMPLEMENT_DYNAMIC(CListCtrlEx, CListCtrl)\n\nCListCtrlEx::CListCtrlEx()\n{\n}\n\n\nCListCtrlEx::~CListCtrlEx()\n{\n}\n\nvoid CListCtrlEx::Edit(int row, int col)\n{\n    EnsureVisible(row, FALSE);\t\t\t\t//༭һʱȷпɼ\n    m_editing = true;\n\n    m_edit_row = row;\n    m_edit_col = col;\n    CRect item_rect;\n    GetSubItemRect(row, col, LVIR_LABEL, item_rect);\t//ȡľ\n\n    CString text = GetItemText(row, col);\t\t//ȡ\n\n    m_item_edit.SetWindowText(text);\t\t//ʾ༭\n    m_item_edit.ShowWindow(SW_SHOW);\t\t//ʾ༭\n    m_item_edit.MoveWindow(item_rect);\t\t//༭ƶ棬\n    m_item_edit.SetFocus();\t\t\t\t\t//ʹ༭ȡý\n    m_item_edit.SetSel(0, -1);\n}\n\nvoid CListCtrlEx::SetEditColMethod(eEditColMethod method)\n{\n    m_edit_col_method = method;\n}\n\nvoid CListCtrlEx::SetEditableCol(const std::initializer_list<int>& paras)\n{\n    m_edit_cols = paras;\n}\n\nvoid CListCtrlEx::EndEdit()\n{\n    if (m_editing)\n    {\n        if (m_edit_row >= 0 && m_edit_row < GetItemCount())\n        {\n            CString str;\n            m_item_edit.GetWindowText(str);\t//ȡñ༭\n            SetItemText(m_edit_row, m_edit_col, str);\t//ݸµCListCtrl\n        }\n        m_item_edit.ShowWindow(SW_HIDE);//ر༭\n        m_editing = false;\n    }\n}\n\nBEGIN_MESSAGE_MAP(CListCtrlEx, CListCtrl)\n    ON_EN_KILLFOCUS(IDC_ITEM_EDITBOX, &CListCtrlEx::OnEnKillfocusEdit1)\n    ON_NOTIFY_REFLECT(NM_DBLCLK, &CListCtrlEx::OnNMDblclk)\n    ON_NOTIFY_REFLECT(LVN_BEGINSCROLL, &CListCtrlEx::OnLvnBeginScroll)\nEND_MESSAGE_MAP()\n\n\nvoid CListCtrlEx::OnEnKillfocusEdit1()\n{\n    //ı༭ؼؼʧȥʱӦ\n    EndEdit();\n}\n\n\n\nvoid CListCtrlEx::PreSubclassWindow()\n{\n    // TODO: ڴרô/û\n    m_item_edit.Create(WS_BORDER | ES_AUTOHSCROLL, CRect(), this, IDC_ITEM_EDITBOX);\n    m_item_edit.SetFont(GetFont());\n\n    CListCtrl::PreSubclassWindow();\n}\n\n\nvoid CListCtrlEx::OnNMDblclk(NMHDR *pNMHDR, LRESULT *pResult)\n{\n    LPNMITEMACTIVATE pNMItemActivate = reinterpret_cast<LPNMITEMACTIVATE>(pNMHDR);\n    // TODO: ڴӿؼ֪ͨ\n\n    //˫Ҫ༭\n    if (m_edit_col_method == EC_ALL\n        || (m_edit_col_method == EC_SPECIFIED && m_edit_cols.find(pNMItemActivate->iSubItem) != m_edit_cols.end()))\n    {\n        Edit(pNMItemActivate->iItem, pNMItemActivate->iSubItem);\n    }\n\n    *pResult = 0;\n}\n\n\nvoid CListCtrlEx::OnLvnBeginScroll(NMHDR *pNMHDR, LRESULT *pResult)\n{\n    // ˹Ҫ Internet Explorer 5.5 ߰汾\n    //  _WIN32_IE  >= 0x0560\n    LPNMLVSCROLL pStateChanged = reinterpret_cast<LPNMLVSCROLL>(pNMHDR);\n    // TODO: ڴӿؼ֪ͨ\n    EndEdit();\n\n    *pResult = 0;\n}\n"
  },
  {
    "path": "TrafficMonitor/ListCtrlEx.h",
    "content": "#pragma once\n#include \"afxcmn.h\"\n#include \"DrawCommon.h\"\n#include <set>\n\n#define IDC_ITEM_EDITBOX 1991\nclass CListCtrlEx :\n\tpublic CListCtrl\n{\nDECLARE_DYNAMIC(CListCtrlEx)\npublic:\n\tCListCtrlEx();\n\t~CListCtrlEx();\n\n    void Edit(int row, int col);\t\t\t//༭ָԪ\n\n    enum eEditColMethod       //Ҫ༭еķʽ\n    {\n        EC_NONE,        //\n        EC_ALL,         //ȫ\n        EC_SPECIFIED    //ָ\n    };\n    void SetEditColMethod(eEditColMethod method);\n    void SetEditableCol(const std::initializer_list<int>& paras);   //༭\n\nprivate:\n    CEdit m_item_edit;\n    int m_edit_row{};\n    int m_edit_col{};\n    bool m_editing{};\n\n    eEditColMethod m_edit_col_method{ EC_NONE };\n    std::set<int> m_edit_cols;\n\nprotected:\n    void EndEdit();\n\n\tDECLARE_MESSAGE_MAP()\n    afx_msg void OnEnKillfocusEdit1();\n    virtual void PreSubclassWindow();\n    afx_msg void OnNMDblclk(NMHDR *pNMHDR, LRESULT *pResult);\n    afx_msg void OnLvnBeginScroll(NMHDR *pNMHDR, LRESULT *pResult);\n};\n\n"
  },
  {
    "path": "TrafficMonitor/MainWndColorDlg.cpp",
    "content": "﻿// MainWndColorDlg.cpp : 实现文件\n//\n\n#include \"stdafx.h\"\n#include \"TrafficMonitor.h\"\n#include \"MainWndColorDlg.h\"\n#include \"afxdialogex.h\"\n#include \"CMFCColorDialogEx.h\"\n\n\n// CMainWndColorDlg 对话框\n\nIMPLEMENT_DYNAMIC(CMainWndColorDlg, CBaseDialog)\n\nCMainWndColorDlg::CMainWndColorDlg(const std::map<CommonDisplayItem, COLORREF>& colors, CWnd* pParent /*=NULL*/)\n\t: CBaseDialog(IDD_MAIN_COLOR_DIALOG, pParent), m_colors(colors)\n{\n}\n\nCMainWndColorDlg::~CMainWndColorDlg()\n{\n}\n\nCString CMainWndColorDlg::GetDialogName() const\n{\n    return _T(\"MainWndColorDlg\");\n}\n\nvoid CMainWndColorDlg::DoDataExchange(CDataExchange* pDX)\n{\n    CBaseDialog::DoDataExchange(pDX);\n    DDX_Control(pDX, IDC_LIST1, m_list_ctrl);\n}\n\n\nBEGIN_MESSAGE_MAP(CMainWndColorDlg, CBaseDialog)\n    ON_NOTIFY(NM_DBLCLK, IDC_LIST1, &CMainWndColorDlg::OnNMDblclkList1)\nEND_MESSAGE_MAP()\n\n\n// CMainWndColorDlg 消息处理程序\n\n\nBOOL CMainWndColorDlg::OnInitDialog()\n{\n\tCBaseDialog::OnInitDialog();\n\n\t// TODO:  在此添加额外的初始化\n\n    SetIcon(theApp.GetMenuIcon(IDI_MAIN_WINDOW), FALSE);\t\t// 设置小图标\n\n    //初始化列表控件\n    CRect rect;\n    m_list_ctrl.GetClientRect(rect);\n    m_list_ctrl.SetExtendedStyle(LVS_EX_FULLROWSELECT | LVS_EX_LABELTIP);\n    int width0, width1;\n    width0 = rect.Width() * 2 / 3;\n    width1 = rect.Width() - width0 - theApp.DPI(20) - 1;\n    m_list_ctrl.InsertColumn(0, CCommon::LoadText(IDS_ITEM), LVCFMT_LEFT, width0);\t\t//插入第0列\n    m_list_ctrl.InsertColumn(1, CCommon::LoadText(IDS_COLOR), LVCFMT_LEFT, width1);\t\t//插入第1列\n    m_list_ctrl.SetDrawItemRangMargin(theApp.DPI(2));\n\n    //向列表中插入行\n    for (auto iter = theApp.m_plugins.AllDisplayItemsWithPlugins().begin(); iter != theApp.m_plugins.AllDisplayItemsWithPlugins().end(); ++iter)\n    {\n        CString item_name = iter->GetItemName();\n        if (!item_name.IsEmpty())\n        {\n            int index = m_list_ctrl.GetItemCount();\n            m_list_ctrl.InsertItem(index, item_name);\n            m_list_ctrl.SetItemColor(index, 1, m_colors[*iter]);\n            m_list_ctrl.SetItemData(index, (DWORD_PTR)&(*iter));\n        }\n    }\n\n\n\treturn TRUE;  // return TRUE unless you set the focus to a control\n\t\t\t\t  // 异常: OCX 属性页应返回 FALSE\n}\n\n\n\nvoid CMainWndColorDlg::OnNMDblclkList1(NMHDR *pNMHDR, LRESULT *pResult)\n{\n    LPNMITEMACTIVATE pNMItemActivate = reinterpret_cast<LPNMITEMACTIVATE>(pNMHDR);\n    // TODO: 在此添加控件通知处理程序代码\n    int index = pNMItemActivate->iItem;\n    COLORREF color = m_list_ctrl.GetItemColor(index, 1);\n    CMFCColorDialogEx colorDlg(color, 0, this);\n    if (colorDlg.DoModal() == IDOK)\n    {\n        color = colorDlg.GetColor();\n        m_list_ctrl.SetItemColor(index, 1, color);\n        CommonDisplayItem* item = (CommonDisplayItem*)(m_list_ctrl.GetItemData(index));\n        m_colors[*item] = color;\n    }\n\n    *pResult = 0;\n}\n"
  },
  {
    "path": "TrafficMonitor/MainWndColorDlg.h",
    "content": "﻿#pragma once\n#include \"ColorStatic.h\"\n#include \"afxwin.h\"\n#include \"ColorSettingListCtrl.h\"\n#include \"BaseDialog.h\"\n\n// CMainWndColorDlg 对话框\n\nclass CMainWndColorDlg : public CBaseDialog\n{\n\tDECLARE_DYNAMIC(CMainWndColorDlg)\n\npublic:\n\tCMainWndColorDlg(const std::map<CommonDisplayItem, COLORREF>& colors, CWnd* pParent = NULL);   // 标准构造函数\n\tvirtual ~CMainWndColorDlg();\n\n    const std::map<CommonDisplayItem, COLORREF>& GetColors() const { return m_colors; }\n\n// 对话框数据\n#ifdef AFX_DESIGN_TIME\n\tenum { IDD = IDD_MAIN_COLOR_DIALOG };\n#endif\nprotected:\n    std::map<CommonDisplayItem, COLORREF> m_colors;\n    CColorSettingListCtrl m_list_ctrl;\n\n    virtual CString GetDialogName() const override;\n\nprotected:\n\tvirtual void DoDataExchange(CDataExchange* pDX);    // DDX/DDV 支持\n\n\tDECLARE_MESSAGE_MAP()\npublic:\n\tvirtual BOOL OnInitDialog();\n    afx_msg void OnNMDblclkList1(NMHDR *pNMHDR, LRESULT *pResult);\n};\n"
  },
  {
    "path": "TrafficMonitor/MainWndSettingsDlg.cpp",
    "content": "﻿// MainWndSettingsDlg.cpp : 实现文件\n//\n\n#include \"stdafx.h\"\n#include \"TrafficMonitor.h\"\n#include \"MainWndSettingsDlg.h\"\n#include \"afxdialogex.h\"\n#include \"CMFCColorDialogEx.h\"\n#include \"DisplayTextSettingDlg.h\"\n\n// CMainWndSettingsDlg 对话框\n\nIMPLEMENT_DYNAMIC(CMainWndSettingsDlg, CTabDlg)\n\nCMainWndSettingsDlg::CMainWndSettingsDlg(CWnd* pParent /*=NULL*/)\n    : CTabDlg(IDD_MAIN_WND_SETTINGS_DIALOG, pParent)\n{\n\n}\n\nCMainWndSettingsDlg::~CMainWndSettingsDlg()\n{\n}\n\nvoid CMainWndSettingsDlg::SetControlMouseWheelEnable(bool enable)\n{\n    m_unit_combo.SetMouseWheelEnable(enable);\n    m_double_click_combo.SetMouseWheelEnable(enable);\n    m_font_size_edit.SetMouseWheelEnable(enable);\n    m_memory_display_combo.SetMouseWheelEnable(enable);\n}\n\nvoid CMainWndSettingsDlg::DrawStaticColor()\n{\n    //CCommon::FillStaticColor(m_color_static, m_data.text_color);\n    if (m_data.text_colors.empty())\n        return;\n    if (m_data.specify_each_item_color)\n    {\n        int color_num{};\n#ifdef WITHOUT_TEMPERATURE\n        color_num = 4;\n#else\n        color_num = 8;\n#endif\n        m_color_static.SetColorNum(color_num);\n        int index{};\n        for (const auto& item : m_data.text_colors)\n        {\n            m_color_static.SetFillColor(index, item.second);\n            index++;\n        }\n        m_color_static.Invalidate();\n    }\n    else\n    {\n        m_color_static.SetFillColor(m_data.text_colors.begin()->second);\n    }\n}\n\nvoid CMainWndSettingsDlg::IniUnitCombo()\n{\n    m_unit_combo.ResetContent();\n    m_unit_combo.AddString(CCommon::LoadText(IDS_AUTO));\n    if (m_data.unit_byte)\n    {\n        m_unit_combo.AddString(CCommon::LoadText(IDS_FIXED_AS, _T(\" KB/s\")));\n        m_unit_combo.AddString(CCommon::LoadText(IDS_FIXED_AS, _T(\" MB/s\")));\n    }\n    else\n    {\n        m_unit_combo.AddString(CCommon::LoadText(IDS_FIXED_AS, _T(\" Kb/s\")));\n        m_unit_combo.AddString(CCommon::LoadText(IDS_FIXED_AS, _T(\" Mb/s\")));\n    }\n    m_unit_combo.SetCurSel(static_cast<int>(m_data.speed_unit));\n}\n\nvoid CMainWndSettingsDlg::EnableControl()\n{\n    bool exe_path_enable = (m_data.double_click_action == DoubleClickAction::SEPCIFIC_APP);\n    CWnd* pWnd{};\n    pWnd = GetDlgItem(IDC_EXE_PATH_STATIC);\n    if (pWnd != nullptr)\n        pWnd->ShowWindow(exe_path_enable ? SW_SHOW : SW_HIDE);\n    pWnd = GetDlgItem(IDC_EXE_PATH_EDIT);\n    if (pWnd != nullptr)\n        pWnd->ShowWindow(exe_path_enable ? SW_SHOW : SW_HIDE);\n    pWnd = GetDlgItem(IDC_BROWSE_BUTTON);\n    if (pWnd != nullptr)\n        pWnd->ShowWindow(exe_path_enable ? SW_SHOW : SW_HIDE);\n}\n\nvoid CMainWndSettingsDlg::DoDataExchange(CDataExchange* pDX)\n{\n    DDX_Control(pDX, IDC_TEXT_COLOR_STATIC, m_color_static);\n    CTabDlg::DoDataExchange(pDX);\n    DDX_Control(pDX, IDC_HIDE_UNIT_CHECK, m_hide_unit_chk);\n    DDX_Control(pDX, IDC_UNIT_COMBO, m_unit_combo);\n    DDX_Control(pDX, IDC_FONT_SIZE_EDIT, m_font_size_edit);\n    DDX_Control(pDX, IDC_DOUBLE_CLICK_COMBO, m_double_click_combo);\n    DDX_Control(pDX, IDC_MEMORY_DISPLAY_COMBO, m_memory_display_combo);\n}\n\n\nBEGIN_MESSAGE_MAP(CMainWndSettingsDlg, CTabDlg)\n    //ON_EN_CHANGE(IDC_UPLOAD_EDIT, &CMainWndSettingsDlg::OnEnChangeUploadEdit)\n    //ON_EN_CHANGE(IDC_DOWNLOAD_EDIT, &CMainWndSettingsDlg::OnEnChangeDownloadEdit)\n    //ON_EN_CHANGE(IDC_CPU_EDIT, &CMainWndSettingsDlg::OnEnChangeCpuEdit)\n    //ON_EN_CHANGE(IDC_MEMORY_EDIT, &CMainWndSettingsDlg::OnEnChangeMemoryEdit)\n    //ON_BN_CLICKED(IDC_SET_COLOR_BUTTON1, &CMainWndSettingsDlg::OnBnClickedSetColorButton1)\n    //ON_BN_CLICKED(IDC_SET_DEFAULT_BUTTON, &CMainWndSettingsDlg::OnBnClickedSetDefaultButton)\n    ON_BN_CLICKED(IDC_SET_FONT_BUTTON, &CMainWndSettingsDlg::OnBnClickedSetFontButton)\n    ON_BN_CLICKED(IDC_SWITCH_UP_DOWN_CHECK, &CMainWndSettingsDlg::OnBnClickedSwitchUpDownCheck)\n    ON_BN_CLICKED(IDC_FULLSCREEN_HIDE_CHECK, &CMainWndSettingsDlg::OnBnClickedFullscreenHideCheck)\n    ON_BN_CLICKED(IDC_SPEED_SHORT_MODE_CHECK2, &CMainWndSettingsDlg::OnBnClickedSpeedShortModeCheck2)\n    ON_CBN_SELCHANGE(IDC_UNIT_COMBO, &CMainWndSettingsDlg::OnCbnSelchangeUnitCombo)\n    ON_BN_CLICKED(IDC_HIDE_UNIT_CHECK, &CMainWndSettingsDlg::OnBnClickedHideUnitCheck)\n    ON_BN_CLICKED(IDC_HIDE_PERCENTAGE_CHECK, &CMainWndSettingsDlg::OnBnClickedHidePercentageCheck)\n    ON_MESSAGE(WM_STATIC_CLICKED, &CMainWndSettingsDlg::OnStaticClicked)\n    ON_BN_CLICKED(IDC_SPECIFY_EACH_ITEM_COLOR_CHECK, &CMainWndSettingsDlg::OnBnClickedSpecifyEachItemColorCheck)\n    ON_CBN_SELCHANGE(IDC_DOUBLE_CLICK_COMBO, &CMainWndSettingsDlg::OnCbnSelchangeDoubleClickCombo)\n    ON_BN_CLICKED(IDC_SEPARATE_VALUE_UNIT_CHECK, &CMainWndSettingsDlg::OnBnClickedSeparateValueUnitCheck)\n    ON_BN_CLICKED(IDC_UNIT_BYTE_RADIO, &CMainWndSettingsDlg::OnBnClickedUnitByteRadio)\n    ON_BN_CLICKED(IDC_UNIT_BIT_RADIO, &CMainWndSettingsDlg::OnBnClickedUnitBitRadio)\n    ON_BN_CLICKED(IDC_SHOW_TOOL_TIP_CHK, &CMainWndSettingsDlg::OnBnClickedShowToolTipChk)\n    ON_BN_CLICKED(IDC_BROWSE_BUTTON, &CMainWndSettingsDlg::OnBnClickedBrowseButton)\n    ON_BN_CLICKED(IDC_DISPLAY_TEXT_SETTING_BUTTON, &CMainWndSettingsDlg::OnBnClickedDisplayTextSettingButton)\n    ON_CBN_SELCHANGE(IDC_MEMORY_DISPLAY_COMBO, &CMainWndSettingsDlg::OnCbnSelchangeMemoryDisplayCombo)\n    ON_BN_CLICKED(IDC_ALWAYS_ON_TOP_CHECK, &CMainWndSettingsDlg::OnBnClickedAlwaysOnTopCheck)\n    ON_BN_CLICKED(IDC_MOUSE_PENETRATE_CHECK, &CMainWndSettingsDlg::OnBnClickedMousePenetrateCheck)\n    ON_BN_CLICKED(IDC_LOCK_WINDOW_POS_CHECK, &CMainWndSettingsDlg::OnBnClickedLockWindowPosCheck)\n    ON_BN_CLICKED(IDC_ALOW_OUT_OF_BORDER_CHECK, &CMainWndSettingsDlg::OnBnClickedAlowOutOfBorderCheck)\nEND_MESSAGE_MAP()\n\n\n// CMainWndSettingsDlg 消息处理程序\n\n\nBOOL CMainWndSettingsDlg::OnInitDialog()\n{\n    CTabDlg::OnInitDialog();\n\n    // TODO:  在此添加额外的初始化\n\n    //初始化各控件状态\n    SetDlgItemText(IDC_FONT_NAME_EDIT, m_data.font.name);\n    //wchar_t buff[16];\n    //swprintf_s(buff, L\"%d\", m_data.font_size);\n    //SetDlgItemText(IDC_FONT_SIZE_EDIT, buff);\n    m_font_size_edit.SetRange(5, 72);\n    m_font_size_edit.SetValue(m_data.font.size);\n\n    //SetDlgItemText(IDC_UPLOAD_EDIT, m_data.disp_str.Get(TDI_UP).c_str());\n    //SetDlgItemText(IDC_DOWNLOAD_EDIT, m_data.disp_str.Get(TDI_DOWN).c_str());\n    //SetDlgItemText(IDC_CPU_EDIT, m_data.disp_str.Get(TDI_CPU).c_str());\n    //SetDlgItemText(IDC_MEMORY_EDIT, m_data.disp_str.Get(TDI_MEMORY).c_str());\n\n    ((CButton*)GetDlgItem(IDC_SWITCH_UP_DOWN_CHECK))->SetCheck(m_data.swap_up_down);\n    ((CButton*)GetDlgItem(IDC_FULLSCREEN_HIDE_CHECK))->SetCheck(m_data.hide_main_wnd_when_fullscreen);\n    ((CButton*)GetDlgItem(IDC_SPEED_SHORT_MODE_CHECK2))->SetCheck(m_data.speed_short_mode);\n    ((CButton*)GetDlgItem(IDC_SEPARATE_VALUE_UNIT_CHECK))->SetCheck(m_data.separate_value_unit_with_space);\n    ((CButton*)GetDlgItem(IDC_SHOW_TOOL_TIP_CHK))->SetCheck(m_data.show_tool_tip);\n\n    m_color_static.SetLinkCursor();\n    DrawStaticColor();\n\n    m_toolTip.Create(this);\n    m_toolTip.SetMaxTipWidth(theApp.DPI(300));\n    m_toolTip.AddTool(GetDlgItem(IDC_SPEED_SHORT_MODE_CHECK2), CCommon::LoadText(IDS_SPEED_SHORT_MODE_TIP));\n\n    if (m_data.unit_byte)\n        ((CButton*)GetDlgItem(IDC_UNIT_BYTE_RADIO))->SetCheck(TRUE);\n    else\n        ((CButton*)GetDlgItem(IDC_UNIT_BIT_RADIO))->SetCheck(TRUE);\n\n    IniUnitCombo();\n\n    m_hide_unit_chk.SetCheck(m_data.hide_unit);\n    if (m_data.speed_unit == SpeedUnit::AUTO)\n    {\n        m_hide_unit_chk.SetCheck(FALSE);\n        m_data.hide_unit = false;\n        m_hide_unit_chk.EnableWindow(FALSE);\n    }\n    ((CButton*)GetDlgItem(IDC_HIDE_PERCENTAGE_CHECK))->SetCheck(m_data.hide_percent);\n\n    if (m_text_disable)\n    {\n        //GetDlgItem(IDC_UPLOAD_EDIT)->EnableWindow(FALSE);\n        //GetDlgItem(IDC_DOWNLOAD_EDIT)->EnableWindow(FALSE);\n        //GetDlgItem(IDC_CPU_EDIT)->EnableWindow(FALSE);\n        //GetDlgItem(IDC_MEMORY_EDIT)->EnableWindow(FALSE);\n        EnableDlgCtrl(IDC_DISPLAY_TEXT_SETTING_BUTTON, false);\n        m_data.swap_up_down = false;\n        ((CButton*)GetDlgItem(IDC_SWITCH_UP_DOWN_CHECK))->SetCheck(FALSE);\n        GetDlgItem(IDC_SWITCH_UP_DOWN_CHECK)->EnableWindow(FALSE);\n        //GetDlgItem(IDC_SET_DEFAULT_BUTTON)->EnableWindow(FALSE);\n    }\n\n    CheckDlgButton(IDC_SPECIFY_EACH_ITEM_COLOR_CHECK, m_data.specify_each_item_color);\n\n    m_double_click_combo.AddString(CCommon::LoadText(IDS_OPEN_CONNECTION_DETIAL));\n    m_double_click_combo.AddString(CCommon::LoadText(IDS_OPEN_HISTORICAL_TRAFFIC));\n    m_double_click_combo.AddString(CCommon::LoadText(IDS_SHOW_HIDE_MORE_INFO));\n    m_double_click_combo.AddString(CCommon::LoadText(IDS_OPEN_OPTION_SETTINGS));\n    m_double_click_combo.AddString(CCommon::LoadText(IDS_OPEN_TASK_MANAGER));\n    m_double_click_combo.AddString(CCommon::LoadText(IDS_SPECIFIC_APP));\n    m_double_click_combo.AddString(CCommon::LoadText(IDS_CHANGE_SKIN));\n    m_double_click_combo.AddString(CCommon::LoadText(IDS_NONE));\n    m_double_click_combo.SetCurSel(static_cast<int>(m_data.double_click_action));\n\n    SetDlgItemText(IDC_EXE_PATH_EDIT, m_data.double_click_exe.c_str());\n    EnableControl();\n\n    //初始化内存显示方式下拉列表\n    m_memory_display_combo.AddString(CCommon::LoadText(IDS_USAGE_PERCENTAGE));\n    m_memory_display_combo.AddString(CCommon::LoadText(IDS_MEMORY_USED));\n    m_memory_display_combo.AddString(CCommon::LoadText(IDS_MEMORY_AVAILABLE));\n    m_memory_display_combo.SetCurSel(static_cast<int>(m_data.memory_display));\n\n    CheckDlgButton(IDC_ALWAYS_ON_TOP_CHECK, m_data.m_always_on_top);\n    CheckDlgButton(IDC_MOUSE_PENETRATE_CHECK, m_data.m_mouse_penetrate);\n    CheckDlgButton(IDC_LOCK_WINDOW_POS_CHECK, m_data.m_lock_window_pos);\n    CheckDlgButton(IDC_ALOW_OUT_OF_BORDER_CHECK, m_data.m_alow_out_of_border);\n\n    ////设置控件不响应鼠标滚轮消息\n    //m_unit_combo.SetMouseWheelEnable(false);\n    //m_double_click_combo.SetMouseWheelEnable(false);\n\n    return TRUE;  // return TRUE unless you set the focus to a control\n                  // 异常: OCX 属性页应返回 FALSE\n}\n\n\n//void CMainWndSettingsDlg::OnEnChangeUploadEdit()\n//{\n//  // TODO:  如果该控件是 RICHEDIT 控件，它将不\n//  // 发送此通知，除非重写 CTabDlg::OnInitDialog()\n//  // 函数并调用 CRichEditCtrl().SetEventMask()，\n//  // 同时将 ENM_CHANGE 标志“或”运算到掩码中。\n//\n//  // TODO:  在此添加控件通知处理程序代码\n//  CString tmp;\n//  GetDlgItemText(IDC_UPLOAD_EDIT, tmp);\n//  m_data.disp_str.Get(TDI_UP) = tmp;\n//}\n//\n//\n//void CMainWndSettingsDlg::OnEnChangeDownloadEdit()\n//{\n//  // TODO:  如果该控件是 RICHEDIT 控件，它将不\n//  // 发送此通知，除非重写 CTabDlg::OnInitDialog()\n//  // 函数并调用 CRichEditCtrl().SetEventMask()，\n//  // 同时将 ENM_CHANGE 标志“或”运算到掩码中。\n//\n//  // TODO:  在此添加控件通知处理程序代码\n//  CString tmp;\n//  GetDlgItemText(IDC_DOWNLOAD_EDIT, tmp);\n//  m_data.disp_str.Get(TDI_DOWN) = tmp;\n//}\n//\n//\n//void CMainWndSettingsDlg::OnEnChangeCpuEdit()\n//{\n//  // TODO:  如果该控件是 RICHEDIT 控件，它将不\n//  // 发送此通知，除非重写 CTabDlg::OnInitDialog()\n//  // 函数并调用 CRichEditCtrl().SetEventMask()，\n//  // 同时将 ENM_CHANGE 标志“或”运算到掩码中。\n//\n//  // TODO:  在此添加控件通知处理程序代码\n//  CString tmp;\n//  GetDlgItemText(IDC_CPU_EDIT, tmp);\n//  m_data.disp_str.Get(TDI_CPU) = tmp;\n//}\n//\n//\n//void CMainWndSettingsDlg::OnEnChangeMemoryEdit()\n//{\n//  // TODO:  如果该控件是 RICHEDIT 控件，它将不\n//  // 发送此通知，除非重写 CTabDlg::OnInitDialog()\n//  // 函数并调用 CRichEditCtrl().SetEventMask()，\n//  // 同时将 ENM_CHANGE 标志“或”运算到掩码中。\n//\n//  // TODO:  在此添加控件通知处理程序代码\n//  CString tmp;\n//  GetDlgItemText(IDC_MEMORY_EDIT, tmp);\n//  m_data.disp_str.Get(TDI_MEMORY) = tmp;\n//}\n\n\n//void CMainWndSettingsDlg::OnBnClickedSetDefaultButton()\n//{\n//  // TODO: 在此添加控件通知处理程序代码\n//  m_data.disp_str.Get(TDI_UP) = CCommon::LoadText(IDS_UPLOAD_DISP, _T(\": \"));\n//  m_data.disp_str.Get(TDI_DOWN) = CCommon::LoadText(IDS_DOWNLOAD_DISP, _T(\": \"));\n//  m_data.disp_str.Get(TDI_CPU) = L\"CPU: \";\n//  m_data.disp_str.Get(TDI_MEMORY) = CCommon::LoadText(IDS_MEMORY_DISP, _T(\": \"));\n//  SetDlgItemText(IDC_UPLOAD_EDIT, m_data.disp_str.Get(TDI_UP).c_str());\n//  SetDlgItemText(IDC_DOWNLOAD_EDIT, m_data.disp_str.Get(TDI_DOWN).c_str());\n//  SetDlgItemText(IDC_CPU_EDIT, m_data.disp_str.Get(TDI_CPU).c_str());\n//  SetDlgItemText(IDC_MEMORY_EDIT, m_data.disp_str.Get(TDI_MEMORY).c_str());\n//}\n\n\nvoid CMainWndSettingsDlg::OnBnClickedSetFontButton()\n{\n    // TODO: 在此添加控件通知处理程序代码\n    LOGFONT lf{};\n    lf.lfHeight = FontSizeToLfHeight(m_data.font.size);\n    lf.lfWeight = (m_data.font.bold ? FW_BOLD : FW_NORMAL);\n    lf.lfItalic = m_data.font.italic;\n    lf.lfUnderline = m_data.font.underline;\n    lf.lfStrikeOut = m_data.font.strike_out;\n    lf.lfPitchAndFamily = DEFAULT_PITCH | FF_SWISS;\n    //wcsncpy_s(lf.lfFaceName, m_data.font.name.GetString(), 32);\n    CCommon::WStringCopy(lf.lfFaceName, 32, m_data.font.name.GetString());\n    CCommon::NormalizeFont(lf);\n    CFontDialog fontDlg(&lf);   //构造字体对话框，初始选择字体为之前字体\n    if (IDOK == fontDlg.DoModal())     // 显示字体对话框\n    {\n        //获取字体信息\n        m_data.font.name = fontDlg.GetFaceName();\n        m_data.font.size = fontDlg.GetSize() / 10;\n        m_data.font.bold = (fontDlg.IsBold() != FALSE);\n        m_data.font.italic = (fontDlg.IsItalic() != FALSE);\n        m_data.font.underline = (fontDlg.IsUnderline() != FALSE);\n        m_data.font.strike_out = (fontDlg.IsStrikeOut() != FALSE);\n        //将字体信息显示出来\n        SetDlgItemText(IDC_FONT_NAME_EDIT, m_data.font.name);\n        wchar_t buff[16];\n        swprintf_s(buff, L\"%d\", m_data.font.size);\n        SetDlgItemText(IDC_FONT_SIZE_EDIT, buff);\n    }\n}\n\n\nvoid CMainWndSettingsDlg::OnBnClickedSwitchUpDownCheck()\n{\n    // TODO: 在此添加控件通知处理程序代码\n    m_data.swap_up_down = (((CButton*)GetDlgItem(IDC_SWITCH_UP_DOWN_CHECK))->GetCheck() != 0);\n}\n\n\nvoid CMainWndSettingsDlg::OnBnClickedFullscreenHideCheck()\n{\n    // TODO: 在此添加控件通知处理程序代码\n    m_data.hide_main_wnd_when_fullscreen = (((CButton*)GetDlgItem(IDC_FULLSCREEN_HIDE_CHECK))->GetCheck() != 0);\n}\n\n\nvoid CMainWndSettingsDlg::OnBnClickedSpeedShortModeCheck2()\n{\n    // TODO: 在此添加控件通知处理程序代码\n    m_data.speed_short_mode = (((CButton*)GetDlgItem(IDC_SPEED_SHORT_MODE_CHECK2))->GetCheck() != 0);\n}\n\n\nvoid CMainWndSettingsDlg::OnCbnSelchangeUnitCombo()\n{\n    // TODO: 在此添加控件通知处理程序代码\n    m_data.speed_unit = static_cast<SpeedUnit>(m_unit_combo.GetCurSel());\n    if (m_data.speed_unit == SpeedUnit::AUTO)\n    {\n        m_hide_unit_chk.SetCheck(FALSE);\n        m_data.hide_unit = false;\n        m_hide_unit_chk.EnableWindow(FALSE);\n    }\n    else\n    {\n        m_hide_unit_chk.EnableWindow(TRUE);\n    }\n}\n\n\nvoid CMainWndSettingsDlg::OnBnClickedHideUnitCheck()\n{\n    // TODO: 在此添加控件通知处理程序代码\n    m_data.hide_unit = (m_hide_unit_chk.GetCheck() != 0);\n}\n\n\nBOOL CMainWndSettingsDlg::PreTranslateMessage(MSG* pMsg)\n{\n    // TODO: 在此添加专用代码和/或调用基类\n    if (pMsg->message == WM_MOUSEMOVE)\n        m_toolTip.RelayEvent(pMsg);\n\n    return CTabDlg::PreTranslateMessage(pMsg);\n}\n\n\nvoid CMainWndSettingsDlg::OnOK()\n{\n    // TODO: 在此添加专用代码和/或调用基类\n    //获取字体设置\n    int font_size;\n    font_size = m_font_size_edit.GetValue();\n    if (font_size > MAX_FONT_SIZE || font_size < MIN_FONT_SIZE)\n    {\n        CString info;\n        info.Format(CCommon::LoadText(IDS_FONT_SIZE_WARNING), MIN_FONT_SIZE, MAX_FONT_SIZE);\n        MessageBox(info, NULL, MB_OK | MB_ICONWARNING);\n    }\n    else\n    {\n        m_data.font.size = font_size;\n    }\n    GetDlgItemText(IDC_FONT_NAME_EDIT, m_data.font.name);\n\n    CTabDlg::OnOK();\n}\n\n\nvoid CMainWndSettingsDlg::OnBnClickedHidePercentageCheck()\n{\n    // TODO: 在此添加控件通知处理程序代码\n    m_data.hide_percent = (((CButton*)GetDlgItem(IDC_HIDE_PERCENTAGE_CHECK))->GetCheck() != 0);\n}\n\n\nafx_msg LRESULT CMainWndSettingsDlg::OnStaticClicked(WPARAM wParam, LPARAM lParam)\n{\n    switch (::GetDlgCtrlID(((CWnd*)wParam)->m_hWnd))\n    {\n    case IDC_TEXT_COLOR_STATIC:\n    {\n        //设置文本颜色\n        if (m_data.specify_each_item_color)\n        {\n            CMainWndColorDlg colorDlg(m_data.text_colors);\n            if (colorDlg.DoModal() == IDOK)\n            {\n                m_data.text_colors = colorDlg.GetColors();\n                DrawStaticColor();\n            }\n        }\n        else if (!m_data.text_colors.empty())\n        {\n            CMFCColorDialogEx colorDlg(m_data.text_colors.begin()->second, 0, this);\n            if (colorDlg.DoModal() == IDOK)\n            {\n                m_data.text_colors.begin()->second = colorDlg.GetColor();\n                DrawStaticColor();\n            }\n        }\n        break;\n    }\n    default:\n        break;\n    }\n    return 0;\n}\n\n\nvoid CMainWndSettingsDlg::OnBnClickedSpecifyEachItemColorCheck()\n{\n    // TODO: 在此添加控件通知处理程序代码\n    m_data.specify_each_item_color = (((CButton*)GetDlgItem(IDC_SPECIFY_EACH_ITEM_COLOR_CHECK))->GetCheck() != 0);\n    DrawStaticColor();\n}\n\n\nvoid CMainWndSettingsDlg::OnCbnSelchangeDoubleClickCombo()\n{\n    // TODO: 在此添加控件通知处理程序代码\n    m_data.double_click_action = static_cast<DoubleClickAction>(m_double_click_combo.GetCurSel());\n    EnableControl();\n}\n\n\nvoid CMainWndSettingsDlg::OnBnClickedSeparateValueUnitCheck()\n{\n    // TODO: 在此添加控件通知处理程序代码\n    m_data.separate_value_unit_with_space = (((CButton*)GetDlgItem(IDC_SEPARATE_VALUE_UNIT_CHECK))->GetCheck() != 0);\n}\n\n\nvoid CMainWndSettingsDlg::OnBnClickedUnitByteRadio()\n{\n    // TODO: 在此添加控件通知处理程序代码\n    m_data.unit_byte = true;\n    IniUnitCombo();\n}\n\n\nvoid CMainWndSettingsDlg::OnBnClickedUnitBitRadio()\n{\n    // TODO: 在此添加控件通知处理程序代码\n    m_data.unit_byte = false;\n    IniUnitCombo();\n}\n\n\nvoid CMainWndSettingsDlg::OnBnClickedShowToolTipChk()\n{\n    // TODO: 在此添加控件通知处理程序代码\n    m_data.show_tool_tip = (((CButton*)GetDlgItem(IDC_SHOW_TOOL_TIP_CHK))->GetCheck() != 0);\n}\n\n\nvoid CMainWndSettingsDlg::OnBnClickedBrowseButton()\n{\n    // TODO: 在此添加控件通知处理程序代码\n    CString szFilter = CCommon::LoadText(IDS_EXE_FILTER);\n    CFileDialog fileDlg(TRUE, NULL, NULL, 0, szFilter, this);\n    if (IDOK == fileDlg.DoModal())\n    {\n        m_data.double_click_exe = fileDlg.GetPathName();\n        SetDlgItemText(IDC_EXE_PATH_EDIT, m_data.double_click_exe.c_str());\n    }\n}\n\n\nvoid CMainWndSettingsDlg::OnBnClickedDisplayTextSettingButton()\n{\n    // TODO: 在此添加控件通知处理程序代码\n    CDisplayTextSettingDlg dlg(m_data.disp_str, true);\n    dlg.DoModal();\n}\n\n\nvoid CMainWndSettingsDlg::OnCbnSelchangeMemoryDisplayCombo()\n{\n    // TODO: 在此添加控件通知处理程序代码\n    m_data.memory_display = static_cast<MemoryDisplay>(m_memory_display_combo.GetCurSel());\n}\n\n\nvoid CMainWndSettingsDlg::OnBnClickedAlwaysOnTopCheck()\n{\n    // TODO: 在此添加控件通知处理程序代码\n    m_data.m_always_on_top = IsDlgButtonChecked(IDC_ALWAYS_ON_TOP_CHECK) != 0;\n}\n\n\nvoid CMainWndSettingsDlg::OnBnClickedMousePenetrateCheck()\n{\n    // TODO: 在此添加控件通知处理程序代码\n    m_data.m_mouse_penetrate = IsDlgButtonChecked(IDC_MOUSE_PENETRATE_CHECK) != 0;\n}\n\n\nvoid CMainWndSettingsDlg::OnBnClickedLockWindowPosCheck()\n{\n    // TODO: 在此添加控件通知处理程序代码\n    m_data.m_lock_window_pos = IsDlgButtonChecked(IDC_LOCK_WINDOW_POS_CHECK) != 0;\n}\n\n\nvoid CMainWndSettingsDlg::OnBnClickedAlowOutOfBorderCheck()\n{\n    // TODO: 在此添加控件通知处理程序代码\n    m_data.m_alow_out_of_border = IsDlgButtonChecked(IDC_ALOW_OUT_OF_BORDER_CHECK) != 0;\n}\n"
  },
  {
    "path": "TrafficMonitor/MainWndSettingsDlg.h",
    "content": "﻿#pragma once\n#include \"ColorStatic.h\"\n#include \"afxwin.h\"\n#include \"SpinEdit.h\"\n#include \"TabDlg.h\"\n#include \"MainWndColorDlg.h\"\n#include \"ComboBox2.h\"\n\n// CMainWndSettingsDlg 对话框\n\nclass CMainWndSettingsDlg : public CTabDlg\n{\n    DECLARE_DYNAMIC(CMainWndSettingsDlg)\n\npublic:\n    CMainWndSettingsDlg(CWnd* pParent = NULL);   // 标准构造函数\n    virtual ~CMainWndSettingsDlg();\n\n    //选项设置数据\n    MainWndSettingData m_data;\n\n    bool m_text_disable{ false };   //如果为true，则不允许设置“显示文本”，并不允许交换上传和下载的位置\n\n// 对话框数据\n#ifdef AFX_DESIGN_TIME\n    enum { IDD = IDD_MAIN_WND_SETTINGS_DIALOG };\n#endif\n\n\nprotected:\n    //控件变量\n    CColorStatic m_color_static;\n    CToolTipCtrl m_toolTip;\n    CComboBox2 m_unit_combo;\n    CButton m_hide_unit_chk;\n    CSpinEdit m_font_size_edit;\n    CComboBox2 m_double_click_combo;\n    CComboBox2 m_memory_display_combo;\n\nprotected:\n    void DrawStaticColor();\n    void IniUnitCombo();\n    void EnableControl();\n\n    virtual void SetControlMouseWheelEnable(bool enable) override;\n\n    virtual void DoDataExchange(CDataExchange* pDX);    // DDX/DDV 支持\n\n    DECLARE_MESSAGE_MAP()\npublic:\n    virtual BOOL OnInitDialog();\n    //afx_msg void OnEnChangeUploadEdit();\n    //afx_msg void OnEnChangeDownloadEdit();\n    //afx_msg void OnEnChangeCpuEdit();\n    //afx_msg void OnEnChangeMemoryEdit();\n    //afx_msg void OnBnClickedSetDefaultButton();\n    afx_msg void OnBnClickedSetFontButton();\n    afx_msg void OnBnClickedSwitchUpDownCheck();\n    afx_msg void OnBnClickedFullscreenHideCheck();\n    afx_msg void OnBnClickedSpeedShortModeCheck2();\n    afx_msg void OnCbnSelchangeUnitCombo();\n    afx_msg void OnBnClickedHideUnitCheck();\n    virtual BOOL PreTranslateMessage(MSG* pMsg);\n    virtual void OnOK();\n    afx_msg void OnBnClickedHidePercentageCheck();\nprotected:\n    afx_msg LRESULT OnStaticClicked(WPARAM wParam, LPARAM lParam);\npublic:\n    afx_msg void OnBnClickedSpecifyEachItemColorCheck();\n    afx_msg void OnCbnSelchangeDoubleClickCombo();\n    afx_msg void OnBnClickedSeparateValueUnitCheck();\n    afx_msg void OnBnClickedUnitByteRadio();\n    afx_msg void OnBnClickedUnitBitRadio();\n    afx_msg void OnBnClickedShowToolTipChk();\n    afx_msg void OnBnClickedBrowseButton();\n    afx_msg void OnBnClickedDisplayTextSettingButton();\n    afx_msg void OnCbnSelchangeMemoryDisplayCombo();\n    afx_msg void OnBnClickedAlwaysOnTopCheck();\n    afx_msg void OnBnClickedMousePenetrateCheck();\n    afx_msg void OnBnClickedLockWindowPosCheck();\n    afx_msg void OnBnClickedAlowOutOfBorderCheck();\n};\n"
  },
  {
    "path": "TrafficMonitor/MessageDlg.cpp",
    "content": "﻿// HelpDlg.cpp : 实现文件\n//\n\n#include \"stdafx.h\"\n#include \"TrafficMonitor.h\"\n#include \"MessageDlg.h\"\n#include \"DrawCommon.h\"\n\n\n// CMessageDlg 对话框\n\n#define MESSAGE_DLG_ICON_SIZE (theApp.DPI(32))\n\nIMPLEMENT_DYNAMIC(CMessageDlg, CDialog)\n\nCMessageDlg::CMessageDlg(CWnd* pParent /*=NULL*/)\n\t: CDialog(IDD_MESSAGE_DIALOG, pParent)\n{\n\n}\n\nCMessageDlg::~CMessageDlg()\n{\n}\n\nvoid CMessageDlg::SetWindowTitle(LPCTSTR str)\n{\n\tm_title = str;\n}\n\nvoid CMessageDlg::SetInfoText(LPCTSTR str)\n{\n\tm_info = str;\n}\n\nvoid CMessageDlg::SetMessageText(LPCTSTR str)\n{\n\tm_message = str;\n}\n\n//void CMessageDlg::SetLinkInfo(LPCTSTR text, LPCTSTR url)\n//{\n//\tm_link_text = text;\n//\tm_link_url = url;\n//}\n\nvoid CMessageDlg::SetMessageIcon(HICON hIcon)\n{\n    m_icon = hIcon;\n}\n\nvoid CMessageDlg::SetInfoStaticSize(int cx)\n{\n    if (m_icon != NULL && m_info_static.GetSafeHwnd() != NULL)\n    {\n        CRect rc_info{ m_rc_info };\n        rc_info.left = m_rc_info.left + MESSAGE_DLG_ICON_SIZE + theApp.DPI(8);\n        if (cx > 0)\n            rc_info.right = cx;\n        m_info_static.MoveWindow(rc_info);\n    }\n}\n\nvoid CMessageDlg::DoDataExchange(CDataExchange* pDX)\n{\n\tCDialog::DoDataExchange(pDX);\n\tDDX_Control(pDX, IDC_HELP_EDIT, m_message_edit);\n\tDDX_Control(pDX, IDC_INFO_STATIC, m_info_static);\n}\n\n\nBEGIN_MESSAGE_MAP(CMessageDlg, CDialog)\n\tON_WM_GETMINMAXINFO()\n\t//ON_NOTIFY(NM_CLICK, IDC_SYSLINK1, &CMessageDlg::OnNMClickSyslink1)\n    ON_WM_PAINT()\n    ON_WM_SIZE()\nEND_MESSAGE_MAP()\n\n\n// CMessageDlg 消息处理程序\n\n\nBOOL CMessageDlg::OnInitDialog()\n{\n\tCDialog::OnInitDialog();\n\n\t// TODO:  在此添加额外的初始化\n\n\tSetIcon(AfxGetApp()->LoadIcon(IDR_MAINFRAME), FALSE);\t\t// 设置小图标\n\n\t//获取初始时窗口的大小\n\tCRect rect;\n\tGetWindowRect(rect);\n\tm_min_size.cx = rect.Width();\n\tm_min_size.cy = rect.Height();\n\n\tSetWindowText(m_title);\n\tm_info_static.SetWindowText(m_info);\n\tm_message_edit.SetWindowText(m_message);\n\n\t//CWnd* pLinkCtrl = GetDlgItem(IDC_SYSLINK1);\n\t//if (pLinkCtrl != nullptr)\n\t//{\n\t//\tpLinkCtrl->ShowWindow(m_show_link_ctrl);\n\t//\tpLinkCtrl->SetWindowText(_T(\"<a>\") + m_link_text + _T(\"</a>\"));\n\t//}\n\n    //设置图标的位置\n    if (m_icon != NULL)\n    {\n        CRect rc_edit;\n        m_message_edit.GetWindowRect(rc_edit);\n        ScreenToClient(rc_edit);\n        m_icon_pos.x = rc_edit.left;\n        m_icon_pos.y = (rc_edit.top - MESSAGE_DLG_ICON_SIZE) / 2;\n\n        m_info_static.GetWindowRect(m_rc_info);\n        ScreenToClient(m_rc_info);\n        SetInfoStaticSize(0);\n    }\n\n\treturn TRUE;  // return TRUE unless you set the focus to a control\n\t\t\t\t  // 异常: OCX 属性页应返回 FALSE\n}\n\n\nvoid CMessageDlg::OnGetMinMaxInfo(MINMAXINFO* lpMMI)\n{\n\t// TODO: 在此添加消息处理程序代码和/或调用默认值\n\t//限制窗口最小大小\n\tlpMMI->ptMinTrackSize.x = m_min_size.cx;\t\t//设置最小宽度\n\tlpMMI->ptMinTrackSize.y = m_min_size.cy;\t\t//设置最小高度\n\n\tCDialog::OnGetMinMaxInfo(lpMMI);\n}\n\n\n//void CMessageDlg::OnNMClickSyslink1(NMHDR *pNMHDR, LRESULT *pResult)\n//{\n//\t// TODO: 在此添加控件通知处理程序代码\n//\tif(!m_link_url.IsEmpty())\n//\t\tShellExecute(NULL, _T(\"open\"), m_link_url, NULL, NULL, SW_SHOW);\t//打开超链接\n//\n//\t*pResult = 0;\n//}\n\n\nvoid CMessageDlg::OnPaint()\n{\n    CPaintDC dc(this); // device context for painting\n                       // TODO: 在此处添加消息处理程序代码\n                       // 不为绘图消息调用 CDialog::OnPaint()\n\n    CDrawCommon draw;\n    draw.Create(&dc, this);\n    draw.DrawIcon(m_icon, m_icon_pos, CSize(MESSAGE_DLG_ICON_SIZE, MESSAGE_DLG_ICON_SIZE));\n}\n\n\nvoid CMessageDlg::OnSize(UINT nType, int cx, int cy)\n{\n    CDialog::OnSize(nType, cx, cy);\n\n    // TODO: 在此处添加消息处理程序代码\n    SetInfoStaticSize(cx);\n}\n"
  },
  {
    "path": "TrafficMonitor/MessageDlg.h",
    "content": "﻿#pragma once\n#include \"afxwin.h\"\n\n// CMessageDlg 对话框\n\nclass CMessageDlg : public CDialog\n{\n\tDECLARE_DYNAMIC(CMessageDlg)\n\npublic:\n\tCMessageDlg(CWnd* pParent = NULL);   // 标准构造函数\n\tvirtual ~CMessageDlg();\n\n\tvoid SetWindowTitle(LPCTSTR str);\n\tvoid SetInfoText(LPCTSTR str);\n\tvoid SetMessageText(LPCTSTR str);\n\t//void ShowLinkStatic(bool show = true) { m_show_link_ctrl = show; }\n\t//void SetLinkInfo(LPCTSTR text, LPCTSTR url);\n    void SetMessageIcon(HICON hIcon);\n\n// 对话框数据\n#ifdef AFX_DESIGN_TIME\n\tenum { IDD = IDD_MESSAGE_DIALOG };\n#endif\n\nprotected:\n\tCEdit m_message_edit;\n\tCSize m_min_size;\t\t//窗口的最小大小\n\tCStatic m_info_static;\n\n\tCString m_title;\n\tCString m_info;\n\tCString m_message;\n\n\t//CString m_link_text;\n\t//CString m_link_url;\n\n    HICON m_icon{};\n    CPoint m_icon_pos{};        //图标的位置\n    CRect m_rc_info{};          //错误信息Static控件的初始区域\n\n\t//bool m_show_link_ctrl{ false };\n\nprotected:\n    void SetInfoStaticSize(int cx);     //如果设置了图标，则需要将错误信息Static控件向右移动一些\n\n\tvirtual void DoDataExchange(CDataExchange* pDX);    // DDX/DDV 支持\n\n\tDECLARE_MESSAGE_MAP()\n\npublic:\n\tvirtual BOOL OnInitDialog();\n\tafx_msg void OnGetMinMaxInfo(MINMAXINFO* lpMMI);\n\t//afx_msg void OnNMClickSyslink1(NMHDR *pNMHDR, LRESULT *pResult);\n    afx_msg void OnPaint();\n    afx_msg void OnSize(UINT nType, int cx, int cy);\n};\n"
  },
  {
    "path": "TrafficMonitor/NetworkInfoDlg.cpp",
    "content": "﻿// CNetworkInfoDlg.cpp : 实现文件\n//\n\n#include \"stdafx.h\"\n#include \"TrafficMonitor.h\"\n#include \"NetworkInfoDlg.h\"\n#include \"afxdialogex.h\"\n\n\n// CNetworkInfoDlg 对话框\n\nIMPLEMENT_DYNAMIC(CNetworkInfoDlg, CBaseDialog)\n\nCNetworkInfoDlg::CNetworkInfoDlg(vector<NetWorkConection>& adapters, MIB_IFROW* pIfRow, int connection_selected, CWnd* pParent /*=NULL*/)\n    : CBaseDialog(IDD_NETWORK_INFO_DIALOG, pParent), m_connections(adapters), m_pIfRow(pIfRow), m_connection_selected(connection_selected)\n{\n    m_current_connection = connection_selected;\n}\n\nCNetworkInfoDlg::~CNetworkInfoDlg()\n{\n}\n\n\nvoid CNetworkInfoDlg::ShowInfo()\n{\n    CString temp;\n    MIB_IFROW& network_info = GetConnectIfTable(m_connection_selected);\n    //接口名\n    m_info_list.SetItemText(0, 1, network_info.wszName);\n    //接口描述\n    m_info_list.SetItemText(1, 1, CCommon::StrToUnicode((const char*)network_info.bDescr).c_str());\n    //连接类型\n    switch (network_info.dwType)\n    {\n    case IF_TYPE_OTHER: temp = CCommon::LoadText(IDS_IF_TYPE_OTHER); break;\n    case IF_TYPE_ETHERNET_CSMACD: temp = CCommon::LoadText(IDS_IF_TYPE_ETHERNET_CSMACD); break;\n    case IF_TYPE_ISO88025_TOKENRING: temp = CCommon::LoadText(IDS_IF_TYPE_ISO88025_TOKENRING); break;\n    case IF_TYPE_FDDI: temp = CCommon::LoadText(IDS_IF_TYPE_FDDI); break;\n    case IF_TYPE_PPP: temp = CCommon::LoadText(IDS_IF_TYPE_PPP); break;\n    case IF_TYPE_SOFTWARE_LOOPBACK: temp = CCommon::LoadText(IDS_IF_TYPE_SOFTWARE_LOOPBACK); break;\n    case IF_TYPE_ATM: temp = CCommon::LoadText(IDS_IF_TYPE_ATM); break;\n    case IF_TYPE_IEEE80211: temp = CCommon::LoadText(IDS_IF_TYPE_IEEE80211); break;\n    case IF_TYPE_TUNNEL: temp = CCommon::LoadText(IDS_IF_TYPE_TUNNEL); break;\n    case IF_TYPE_IEEE1394: temp = CCommon::LoadText(IDS_IF_TYPE_IEEE1394); break;\n    case IF_TYPE_IEEE80216_WMAN: temp = CCommon::LoadText(IDS_IF_TYPE_IEEE80216_WMAN); break;\n    case IF_TYPE_WWANPP: temp = CCommon::LoadText(IDS_IF_TYPE_WWANPP); break;\n    case IF_TYPE_WWANPP2: temp = CCommon::LoadText(IDS_IF_TYPE_WWANPP2); break;\n    default: temp = CCommon::LoadText(IDS_UNKNOW_CONNECTION); break;\n    }\n    m_info_list.SetItemText(2, 1, temp);\n    //速度\n    temp.Format(_T(\"%dMbps\"), network_info.dwSpeed / 1000000);\n    m_info_list.SetItemText(3, 1, temp);\n    //适配器物理地址\n    temp = _T(\"\");\n    char buff[3];\n    for (size_t i{}; i < network_info.dwPhysAddrLen; i++)\n    {\n        sprintf_s(buff, \"%.2x\", network_info.bPhysAddr[i]);\n        temp += buff;\n        if (i != network_info.dwPhysAddrLen - 1)\n            temp += _T('-');\n    }\n    m_info_list.SetItemText(4, 1, temp);\n    //IP地址\n    m_info_list.SetItemText(5, 1, GetConnection(m_connection_selected).ip_address.c_str());\n    //子网掩码\n    m_info_list.SetItemText(6, 1, GetConnection(m_connection_selected).subnet_mask.c_str());\n    //默认网关\n    m_info_list.SetItemText(7, 1, GetConnection(m_connection_selected).default_gateway.c_str());\n    //连接状态\n    switch (network_info.dwOperStatus)\n    {\n    case IF_OPER_STATUS_NON_OPERATIONAL: temp = CCommon::LoadText(IDS_IF_OPER_STATUS_NON_OPERATIONAL); break;\n    case IF_OPER_STATUS_UNREACHABLE: temp = CCommon::LoadText(IDS_IF_OPER_STATUS_UNREACHABLE); break;\n    case IF_OPER_STATUS_DISCONNECTED: temp = CCommon::LoadText(IDS_IF_OPER_STATUS_DISCONNECTED); break;\n    case IF_OPER_STATUS_CONNECTING: temp = CCommon::LoadText(IDS_IF_OPER_STATUS_CONNECTING); break;\n    case IF_OPER_STATUS_CONNECTED: temp = CCommon::LoadText(IDS_IF_OPER_STATUS_CONNECTED); break;\n    case IF_OPER_STATUS_OPERATIONAL: temp = CCommon::LoadText(IDS_IF_OPER_STATUS_OPERATIONAL); break;\n        //case IfOperStatusUp:\n        //    temp = CCommon::LoadText(IDS_IF_OPER_STATUS_UP);\n        //    break;\n        //case IfOperStatusDown:\n        //case IfOperStatusNotPresent:\n        //case IfOperStatusLowerLayerDown:\n        //    temp = CCommon::LoadText(IDS_IF_OPER_STATUS_DOWN);\n        //    break;\n        //case IfOperStatusTesting:\n        //case IfOperStatusUnknown:\n        //    temp = CCommon::LoadText(IDS_UNKNOW_STATUS);\n        //    break;\n        //case IfOperStatusDormant:\n        //    temp = CCommon::LoadText(IDS_IF_OPER_STATUS_DORMANT);\n            //break;\n    default: temp = CCommon::LoadText(IDS_UNKNOW_STATUS); break;\n    }\n    m_info_list.SetItemText(8, 1, temp);\n    //已接收字节数\n    temp.Format(_T(\"%s (%s)\"), CCommon::IntToString(network_info.dwInOctets, true, true), CCommon::DataSizeToString(network_info.dwInOctets));\n    m_info_list.SetItemText(9, 1, temp);\n    //已发送字节数\n    temp.Format(_T(\"%s (%s)\"), CCommon::IntToString(network_info.dwOutOctets, true, true), CCommon::DataSizeToString(network_info.dwOutOctets));\n    m_info_list.SetItemText(10, 1, temp);\n    //自程序启动以来已接收字节数\n    unsigned __int64 in_bytes_since_start;\n    in_bytes_since_start = network_info.dwInOctets - GetConnection(m_connection_selected).in_bytes;\n    temp.Format(_T(\"%s (%s)\"), CCommon::IntToString(in_bytes_since_start, true, true), CCommon::DataSizeToString(in_bytes_since_start));\n    m_info_list.SetItemText(11, 1, temp);\n    //自程序启动以来已发送字节数\n    unsigned __int64 out_bytes_since_start;\n    out_bytes_since_start = network_info.dwOutOctets - GetConnection(m_connection_selected).out_bytes;\n    temp.Format(_T(\"%s (%s)\"), CCommon::IntToString(out_bytes_since_start, true, true), CCommon::DataSizeToString(out_bytes_since_start));\n    m_info_list.SetItemText(12, 1, temp);\n\n    //显示当前选择指示\n    CString str;\n    str.Format(_T(\"%d/%d\"), m_connection_selected + 1, m_connections.size());\n    SetDlgItemText(IDC_INDEX_STATIC, str);\n    CFont* font = GetFont();\n    CWnd* index_static = GetDlgItem(IDC_INDEX_STATIC);\n    if (m_current_connection == m_connection_selected && !theApp.m_cfg_data.m_select_all)\n        index_static->SetFont(&m_font_bold);\n    else\n        index_static->SetFont(font);\n}\n\nvoid CNetworkInfoDlg::GetProgramElapsedTime()\n{\n    //程序已运行时间\n    SYSTEMTIME current_time, time;\n    GetLocalTime(&current_time);\n    time = CCommon::CompareSystemTime(current_time, m_start_time);\n    CString temp;\n    temp.Format(CCommon::LoadText(IDS_HOUR_MINUTE_SECOND), time.wHour, time.wMinute, time.wSecond);\n    m_info_list.SetItemText(13, 1, temp);\n}\n\nMIB_IFROW& CNetworkInfoDlg::GetConnectIfTable(int connection_index)\n{\n    static MIB_IFROW nouse{};\n    if (connection_index >= 0 && connection_index < static_cast<int>(m_connections.size()))\n    {\n        int index = m_connections[connection_index].index;\n        if (m_pIfRow != nullptr)\n            return m_pIfRow[index];\n    }\n    return nouse;\n}\n\nNetWorkConection CNetworkInfoDlg::GetConnection(int connection_index)\n{\n    if (connection_index >= 0 && connection_index < static_cast<int>(m_connections.size()))\n        return m_connections[connection_index];\n    else\n        return NetWorkConection();\n}\n\nUINT CNetworkInfoDlg::GetInternetIPThreadFunc(LPVOID lpParam)\n{\n    CCommon::SetThreadLanguage(theApp.m_general_data.language);\t\t//设置线程语言\n    CNetworkInfoDlg* p_instance = (CNetworkInfoDlg*)lpParam;\n    wstring ip_address, ip_location;\n\n    //IPV4\n    CCommon::GetInternetIp2(ip_address, ip_location, false);\t\t\t//获取外网IP地址，\n    if (!IsWindow(p_instance->GetSafeHwnd()))\t\t//如果当前对话框已经销毁，则退出线程\n        return 0;\n    if (!ip_address.empty())\n    {\n        CString info;\n        if (ip_location.empty())\n            info = ip_address.c_str();\n        else\n            info.Format(_T(\"%s (%s)\"), ip_address.c_str(), ip_location.c_str());\n        p_instance->m_info_list.SetItemText(14, 1, info);\n    }\n    else\n    {\n        p_instance->m_info_list.SetItemText(14, 1, CCommon::LoadText(IDS_GET_FAILED));\n    }\n\n    //IPV6\n    wstring ipv6_address, ipv6_location;\n    CCommon::GetInternetIp2(ip_address, ip_location, true);\t\t\t//获取外网IP地址，\n    if (!IsWindow(p_instance->GetSafeHwnd()))\t\t//如果当前对话框已经销毁，则退出线程\n        return 0;\n    if (!ip_address.empty())\n    {\n        CString info;\n        if (ip_location.empty())\n            info = ip_address.c_str();\n        else\n            info.Format(_T(\"%s (%s)\"), ip_address.c_str(), ip_location.c_str());\n        p_instance->m_info_list.SetItemText(15, 1, info);\n    }\n    else\n    {\n        p_instance->m_info_list.SetItemText(15, 1, CCommon::LoadText(IDS_GET_FAILED));\n    }\n\n    p_instance->m_ip_acquired = true;\n    return 0;\n}\n\nCString CNetworkInfoDlg::GetDialogName() const\n{\n    return _T(\"NetworkInfoDlg\");\n}\n\nvoid CNetworkInfoDlg::DoDataExchange(CDataExchange* pDX)\n{\n    CBaseDialog::DoDataExchange(pDX);\n    DDX_Control(pDX, IDC_INFO_LIST1, m_info_list);\n}\n\n\nBEGIN_MESSAGE_MAP(CNetworkInfoDlg, CBaseDialog)\n    ON_COMMAND(ID_COPY_TEXT, &CNetworkInfoDlg::OnCopyText)\n    ON_NOTIFY(NM_RCLICK, IDC_INFO_LIST1, &CNetworkInfoDlg::OnNMRClickInfoList1)\n    ON_WM_CLOSE()\n    ON_BN_CLICKED(IDC_PREVIOUS_BUTTON, &CNetworkInfoDlg::OnBnClickedPreviousButton)\n    ON_BN_CLICKED(IDC_NEXT_BUTTON, &CNetworkInfoDlg::OnBnClickedNextButton)\n    ON_WM_TIMER()\n    ON_WM_MOUSEWHEEL()\n    ON_NOTIFY(NM_DBLCLK, IDC_INFO_LIST1, &CNetworkInfoDlg::OnNMDblclkInfoList1)\nEND_MESSAGE_MAP()\n\n\n// CNetworkInfoDlg 消息处理程序\n\n\nBOOL CNetworkInfoDlg::OnInitDialog()\n{\n    CBaseDialog::OnInitDialog();\n\n    // TODO:  在此添加额外的初始化\n    SetWindowText(CCommon::LoadText(IDS_TITLE_CONNECTION_DETIAL));\n    SetIcon(theApp.GetMenuIcon(IDI_INFO), FALSE);\t\t// 设置小图标\n\n    //重新获取IP地址\n    CAdapterCommon::RefreshIpAddress(m_connections);\n\n    //初始化列表控件\n    CRect rect;\n    m_info_list.GetClientRect(rect);\n    m_info_list.SetExtendedStyle(LVS_EX_FULLROWSELECT | LVS_EX_LABELTIP);\n    int width0, width1;\n    width0 = rect.Width() / 4;\n    width1 = rect.Width() - width0 - theApp.DPI(20) - 1;\n    m_info_list.InsertColumn(0, CCommon::LoadText(IDS_ITEM), LVCFMT_LEFT, width0);\t\t//插入第0列\n    m_info_list.InsertColumn(1, CCommon::LoadText(IDS_VALUE), LVCFMT_LEFT, width1);\t\t//插入第1列\n\n    //向列表中插入行\n    m_info_list.InsertItem(0, CCommon::LoadText(IDS_INTERFACE_NAME));\n    m_info_list.InsertItem(1, CCommon::LoadText(IDS_INTERFACE_DESCRIPTION));\n    m_info_list.InsertItem(2, CCommon::LoadText(IDS_CONNECTION_TYPE));\n    m_info_list.InsertItem(3, CCommon::LoadText(IDS_SPEED));\n    m_info_list.InsertItem(4, CCommon::LoadText(IDS_ADAPTER_PHYSICAL_ADDRESS));\n    m_info_list.InsertItem(5, CCommon::LoadText(IDS_IP_ADDRESS));\n    m_info_list.InsertItem(6, CCommon::LoadText(IDS_SUBNET_MASK));\n    m_info_list.InsertItem(7, CCommon::LoadText(IDS_DEFAULT_GATEWAY));\n    m_info_list.InsertItem(8, CCommon::LoadText(IDS_OPERATIONAL_STATUS));\n    m_info_list.InsertItem(9, CCommon::LoadText(IDS_BYTES_RECEIVED));\n    m_info_list.InsertItem(10, CCommon::LoadText(IDS_BYTES_SENT));\n    m_info_list.InsertItem(11, CCommon::LoadText(IDS_BYTES_RECEIVED_SINCE_START));\n    m_info_list.InsertItem(12, CCommon::LoadText(IDS_BYTES_SENT_SINCE_START));\n    m_info_list.InsertItem(13, CCommon::LoadText(IDS_PROGRAM_ELAPSED_TIME));\n    m_info_list.InsertItem(14, CCommon::LoadText(IDS_INTERNET_IP_ADDRESS, _T(\" (ipv4)\")));\n    m_info_list.InsertItem(15, CCommon::LoadText(IDS_INTERNET_IP_ADDRESS, _T(\" (ipv6)\")));\n    //if (theApp.m_cfg_data.m_show_internet_ip)\n    //{\n    //\tm_info_list.SetItemText(14, 1, CCommon::LoadText(IDS_ACQUIRING, _T(\"...\")));\n    //\tm_info_list.SetItemText(15, 1, CCommon::LoadText(IDS_ACQUIRING, _T(\"...\")));\n    //}\n    //else\n    //{\n    m_info_list.SetItemText(14, 1, CCommon::LoadText(IDS_DOUBLE_CLICK_TO_ACQUIRE));\n    m_info_list.SetItemText(15, 1, CCommon::LoadText(IDS_DOUBLE_CLICK_TO_ACQUIRE));\n    //}\n\n    //显示列表中的信息\n    LOGFONT lf{};\n    GetFont()->GetLogFont(&lf);\n    lf.lfWeight = FW_BOLD;\n    m_font_bold.CreateFontIndirect(&lf);\n    ShowInfo();\n    GetProgramElapsedTime();\n\n    //CCommon::GetInternetIp();\n    //if (theApp.m_cfg_data.m_show_internet_ip)\n //       m_pGetIPThread = AfxBeginThread(GetInternetIPThreadFunc, this);\t\t//启动获取外网IP的线程\n\n    //SetWindowPos(&wndNoTopMost, 0, 0, 0, 0, SWP_NOSIZE | SWP_NOMOVE);\t\t//取消置顶\n    m_info_list.GetToolTips()->SetWindowPos(&wndTopMost, 0, 0, 0, 0, SWP_NOSIZE | SWP_NOMOVE);\n\n    m_menu.LoadMenu(IDR_INFO_MENU); //装载右键菜单\n\n    SetTimer(CONNECTION_DETAIL_TIMER, 1000, NULL);\n\n    return TRUE;  // return TRUE unless you set the focus to a control\n                  // 异常: OCX 属性页应返回 FALSE\n}\n\n\nvoid CNetworkInfoDlg::OnCopyText()\n{\n    // TODO: 在此添加命令处理程序代码\n    if (!CCommon::CopyStringToClipboard(wstring(m_selected_string)))\n        MessageBox(CCommon::LoadText(IDS_COPY_TO_CLIPBOARD_FAILED), NULL, MB_ICONWARNING);\n}\n\n\nvoid CNetworkInfoDlg::OnNMRClickInfoList1(NMHDR* pNMHDR, LRESULT* pResult)\n{\n    LPNMITEMACTIVATE pNMItemActivate = reinterpret_cast<LPNMITEMACTIVATE>(pNMHDR);\n    // TODO: 在此添加控件通知处理程序代码\n\n    //获取鼠标点击处的文本\n    int item, sub_item;\n    item = pNMItemActivate->iItem;\n    sub_item = pNMItemActivate->iSubItem;\n    m_selected_string = m_info_list.GetItemText(item, sub_item);\n\n    //弹出右键菜单\n    CMenu* pContextMenu = m_menu.GetSubMenu(0);\t//获取第一个弹出菜单\n    CPoint point1;\t//定义一个用于确定光标位置的位置\n    GetCursorPos(&point1);\t//获取当前光标的位置，以便使得菜单可以跟随光标\n    pContextMenu->TrackPopupMenu(TPM_LEFTALIGN | TPM_RIGHTBUTTON, point1.x, point1.y, this); //在指定位置显示弹出菜单\n\n    *pResult = 0;\n}\n\n\nvoid CNetworkInfoDlg::OnClose()\n{\n    // TODO: 在此添加消息处理程序代码和/或调用默认值\n    //对话框关闭时强制结束获取IP地址的线程\n    //if(theApp.m_cfg_data.m_show_internet_ip)\n //       TerminateThread(m_pGetIPThread->m_hThread, 0);\n    CBaseDialog::OnClose();\n}\n\n\nvoid CNetworkInfoDlg::OnBnClickedPreviousButton()\n{\n    // TODO: 在此添加控件通知处理程序代码\n    if (m_connections.size() > 1 && m_connection_selected > 0)\n    {\n        m_connection_selected--;\n        ShowInfo();\n    }\n}\n\n\nvoid CNetworkInfoDlg::OnBnClickedNextButton()\n{\n    // TODO: 在此添加控件通知处理程序代码\n    if (m_connections.size() > 1 && m_connection_selected < m_connections.size() - 1)\n    {\n        m_connection_selected++;\n        ShowInfo();\n    }\n}\n\n\nBOOL CNetworkInfoDlg::PreTranslateMessage(MSG* pMsg)\n{\n    // TODO: 在此添加专用代码和/或调用基类\n    if (pMsg->message == WM_KEYDOWN)\n    {\n        if (pMsg->wParam == VK_LEFT)\n        {\n            OnBnClickedPreviousButton();\n            return TRUE;\n        }\n        if (pMsg->wParam == VK_RIGHT)\n        {\n            OnBnClickedNextButton();\n            return TRUE;\n        }\n    }\n\n    return CBaseDialog::PreTranslateMessage(pMsg);\n}\n\n\n\nvoid CNetworkInfoDlg::OnTimer(UINT_PTR nIDEvent)\n{\n    // TODO: 在此添加消息处理程序代码和/或调用默认值\n    if (nIDEvent == CONNECTION_DETAIL_TIMER)\n    {\n        GetProgramElapsedTime();\n    }\n\n    CBaseDialog::OnTimer(nIDEvent);\n}\n\nBOOL CNetworkInfoDlg::OnMouseWheel(UINT nFlags, short zDelta, CPoint pt)\n{\n    // TODO: 在此添加消息处理程序代码和/或调用默认值\n    //通过鼠标滚轮翻页\n    if (zDelta > 0)\n    {\n        OnBnClickedPreviousButton();\n    }\n    if (zDelta < 0)\n    {\n        OnBnClickedNextButton();\n    }\n\n    return CBaseDialog::OnMouseWheel(nFlags, zDelta, pt);\n}\n\n\nvoid CNetworkInfoDlg::OnNMDblclkInfoList1(NMHDR* pNMHDR, LRESULT* pResult)\n{\n    LPNMITEMACTIVATE pNMItemActivate = reinterpret_cast<LPNMITEMACTIVATE>(pNMHDR);\n    // TODO: 在此添加控件通知处理程序代码\n    if (/*!theApp.m_cfg_data.m_show_internet_ip && */!m_ip_acquired && (pNMItemActivate->iItem == 14 || pNMItemActivate->iItem == 15))\t\t//双击了IP地址一行时\n    {\n        m_info_list.SetItemText(14, 1, CCommon::LoadText(IDS_ACQUIRING, _T(\"...\")));\n        m_info_list.SetItemText(15, 1, CCommon::LoadText(IDS_ACQUIRING, _T(\"...\")));\n        m_pGetIPThread = AfxBeginThread(GetInternetIPThreadFunc, this);\n    }\n    *pResult = 0;\n}\n"
  },
  {
    "path": "TrafficMonitor/NetworkInfoDlg.h",
    "content": "﻿#pragma once\n#include\"Common.h\"\n#include \"afxcmn.h\"\n#include \"AdapterCommon.h\"\n#include \"BaseDialog.h\"\n\n// CNetworkInfoDlg 对话框\n\nclass CNetworkInfoDlg : public CBaseDialog\n{\n    DECLARE_DYNAMIC(CNetworkInfoDlg)\n\npublic:\n    CNetworkInfoDlg(vector<NetWorkConection>& adapters, MIB_IFROW* pIfRow, int connection_selected, CWnd* pParent = NULL);   // 标准构造函数\n    virtual ~CNetworkInfoDlg();\n\n    // 对话框数据\n#ifdef AFX_DESIGN_TIME\n    enum { IDD = IDD_NETWORK_INFO_DIALOG };\n#endif\n\n    SYSTEMTIME m_start_time;\t\t//程序启动的时间\n\nprotected:\n\n    vector<NetWorkConection>& m_connections;\n    MIB_IFROW* m_pIfRow;\n    int m_connection_selected;\t\t//当前对话框显示的连接\n    int m_current_connection;\t\t//初始选择的连接\n\n    CListCtrl m_info_list;\n    CMenu m_menu;\n    CString m_selected_string;\n    CFont m_font_bold;\t\t//默认字体的粗体\n\n    CWinThread* m_pGetIPThread;\t\t\t//获取外网IP的线程\n    bool m_ip_acquired{ false };        //如果已获取外网ip地址，则为true\n\n    //void GetIPAddress();\t//获取IP地址\n    void ShowInfo();\n    void GetProgramElapsedTime();\n    MIB_IFROW& GetConnectIfTable(int connection_index);    //获取当前选择的网络连接的MIB_IFROW对象。connection_index为m_connections中的索引\n    NetWorkConection GetConnection(int connection_index); //获取当前选择的网络连接的NetWorkConection对象。connection_index为m_connections中的索引\n\n    //获取外网IP的线程函数\n    static UINT GetInternetIPThreadFunc(LPVOID lpParam);\n\n    virtual CString GetDialogName() const override;\n\n    virtual void DoDataExchange(CDataExchange* pDX);    // DDX/DDV 支持\n\n    DECLARE_MESSAGE_MAP()\n\n    virtual BOOL OnInitDialog();\npublic:\n    afx_msg void OnCopyText();\n    afx_msg void OnNMRClickInfoList1(NMHDR* pNMHDR, LRESULT* pResult);\n    afx_msg void OnClose();\n    afx_msg void OnBnClickedPreviousButton();\n    afx_msg void OnBnClickedNextButton();\n    virtual BOOL PreTranslateMessage(MSG* pMsg);\n    afx_msg void OnTimer(UINT_PTR nIDEvent);\n    afx_msg BOOL OnMouseWheel(UINT nFlags, short zDelta, CPoint pt);\n    afx_msg void OnNMDblclkInfoList1(NMHDR* pNMHDR, LRESULT* pResult);\n};\n"
  },
  {
    "path": "TrafficMonitor/OptionsDlg.cpp",
    "content": "﻿// OptionsDlg.cpp : 实现文件\n//\n\n#include \"stdafx.h\"\n#include \"TrafficMonitor.h\"\n#include \"OptionsDlg.h\"\n#include \"afxdialogex.h\"\n\n\n// COptionsDlg 对话框\n\nIMPLEMENT_DYNAMIC(COptionsDlg, CBaseDialog)\n\nCOptionsDlg::COptionsDlg(int tab, CWnd* pParent /*=NULL*/)\n    : CBaseDialog(IDD_OPTIONS_DIALOG, pParent), m_tab_selected{ tab }\n{\n\n}\n\nCOptionsDlg::~COptionsDlg()\n{\n}\n\nCString COptionsDlg::GetDialogName() const\n{\n    return OPTION_DLG_NAME;\n}\n\nvoid COptionsDlg::DoDataExchange(CDataExchange* pDX)\n{\n    CBaseDialog::DoDataExchange(pDX);\n    DDX_Control(pDX, IDC_TAB1, m_tab);\n}\n\n\nBEGIN_MESSAGE_MAP(COptionsDlg, CBaseDialog)\n    ON_WM_SIZE()\n    ON_BN_CLICKED(IDC_APPLY_BUTTON, &COptionsDlg::OnBnClickedApplyButton)\nEND_MESSAGE_MAP()\n\n\n// COptionsDlg 消息处理程序\n\n\nBOOL COptionsDlg::OnInitDialog()\n{\n    CBaseDialog::OnInitDialog();\n\n    // TODO:  在此添加额外的初始化\n    SetWindowText(CCommon::LoadText(IDS_TITLE_OPTION));\n    SetIcon(theApp.GetMenuIcon(IDI_SETTINGS), FALSE);       // 设置小图标\n\n    //创建子对话框\n    m_tab1_dlg.Create(IDD_MAIN_WND_SETTINGS_DIALOG, &m_tab);\n    m_tab2_dlg.Create(IDD_TASKBAR_SETTINGS_DIALOG, &m_tab);\n    m_tab3_dlg.Create(IDD_GENERAL_SETTINGS_DIALOG, &m_tab);\n\n    //保存子对话框\n    m_tab_vect.push_back(&m_tab1_dlg);\n    m_tab_vect.push_back(&m_tab2_dlg);\n    m_tab_vect.push_back(&m_tab3_dlg);\n\n    //获取子对话框的初始高度\n    for (const auto* pDlg : m_tab_vect)\n    {\n        CRect rect;\n        pDlg->GetWindowRect(rect);\n        m_tab_height.push_back(rect.Height());\n    }\n\n    //添加对话框\n    m_tab.AddWindow(&m_tab1_dlg, CCommon::LoadText(IDS_MAIN_WINDOW_SETTINGS));\n    m_tab.AddWindow(&m_tab2_dlg, CCommon::LoadText(IDS_TASKBAR_WINDOW_SETTINGS));\n    m_tab.AddWindow(&m_tab3_dlg, CCommon::LoadText(IDS_GENERAL_SETTINGS));\n\n    //为每个子窗口设置滚动信息\n    for (size_t i = 0; i < m_tab_vect.size(); i++)\n    {\n        m_tab_vect[i]->SetScrollbarInfo(m_tab.m_tab_rect.Height(), m_tab_height[i]);\n    }\n\n    //设置默认选中的标签\n    if (m_tab_selected < 0 || m_tab_selected >= m_tab.GetItemCount())\n        m_tab_selected = 0;\n    m_tab.SetCurTab(m_tab_selected);\n\n    return TRUE;  // return TRUE unless you set the focus to a control\n                  // 异常: OCX 属性页应返回 FALSE\n}\n\n\nvoid COptionsDlg::OnOK()\n{\n    // TODO: 在此添加专用代码和/或调用基类\n    m_tab1_dlg.OnOK();\n    m_tab2_dlg.OnOK();\n    m_tab3_dlg.OnOK();\n\n    CBaseDialog::OnOK();\n}\n\n\nvoid COptionsDlg::OnSize(UINT nType, int cx, int cy)\n{\n    CBaseDialog::OnSize(nType, cx, cy);\n\n    // TODO: 在此处添加消息处理程序代码\n    if (nType != SIZE_MINIMIZED)\n    {\n        //为每个子窗口设置滚动信息\n        for (size_t i = 0; i < m_tab_vect.size(); i++)\n        {\n            m_tab_vect[i]->ResetScroll();\n            m_tab_vect[i]->SetScrollbarInfo(m_tab.m_tab_rect.Height(), m_tab_height[i]);\n        }\n    }\n}\n\n\nvoid COptionsDlg::OnCancel()\n{\n    // TODO: 在此添加专用代码和/或调用基类\n    m_tab3_dlg.OnCancel();\n\n    CBaseDialog::OnCancel();\n}\n\n\nvoid COptionsDlg::OnBnClickedApplyButton()\n{\n    m_tab2_dlg.SaveColorSettingToDefaultStyle();\n    ::SendMessage(theApp.m_pMainWnd->GetSafeHwnd(), WM_SETTINGS_APPLIED, (WPARAM)this, 0);\n}\n"
  },
  {
    "path": "TrafficMonitor/OptionsDlg.h",
    "content": "﻿#pragma once\n#include \"MainWndSettingsDlg.h\"\n#include \"TaskBarSettingsDlg.h\"\n#include \"GeneralSettingsDlg.h\"\n#include \"afxcmn.h\"\n#include \"CTabCtrlEx.h\"\n#include \"BaseDialog.h\"\n\n// COptionsDlg 对话框\n\n#define OPTION_DLG_NAME _T(\"OptionsDlg\")\n\nclass COptionsDlg : public CBaseDialog\n{\n    DECLARE_DYNAMIC(COptionsDlg)\n\npublic:\n    COptionsDlg(int tab = 0, CWnd* pParent = NULL);   // 标准构造函数\n    virtual ~COptionsDlg();\n\n    // 对话框数据\n#ifdef AFX_DESIGN_TIME\n    enum { IDD = IDD_OPTIONS_DIALOG };\n#endif\n\n    CMainWndSettingsDlg m_tab1_dlg{ this };\n    CTaskBarSettingsDlg m_tab2_dlg{ this };\n    CGeneralSettingsDlg m_tab3_dlg{ this };\n\nprotected:\n    CTabCtrlEx m_tab;\n    int m_tab_selected;\n    std::vector<CTabDlg*> m_tab_vect;\n    std::vector<int> m_tab_height;\n\n    virtual CString GetDialogName() const override;\n    virtual void DoDataExchange(CDataExchange* pDX);    // DDX/DDV 支持\n\n    DECLARE_MESSAGE_MAP()\npublic:\n    virtual BOOL OnInitDialog();\n    virtual void OnOK();\n    afx_msg void OnSize(UINT nType, int cx, int cy);\n    virtual void OnCancel();\n    afx_msg void OnBnClickedApplyButton();\n};\n"
  },
  {
    "path": "TrafficMonitor/PictureStatic.cpp",
    "content": "#include \"stdafx.h\"\n#include \"PictureStatic.h\"\n\n\nCPictureStatic::CPictureStatic()\n{\n}\n\n\nCPictureStatic::~CPictureStatic()\n{\n\tm_memDC.DeleteDC();\n\tm_bitmap.Detach();\n}\n\nvoid CPictureStatic::SetPicture(UINT pic_id)\n{\n\tm_memDC.DeleteDC();\n\tif (!m_bitmap.LoadBitmap(pic_id))\t\t//λͼ\n\t\treturn;\n\t//ȡͼʵʴС\n\tGetObject(m_bitmap, sizeof(BITMAP), &m_bm);\n\tCDC* pDC = GetDC();\n\tm_memDC.CreateCompatibleDC(pDC);\n\tm_memDC.SelectObject(&m_bitmap);\n\t//ȡؼС\n\tGetClientRect(m_rect);\n\t//ֶػ\n\tInvalidate();\n}\n\nvoid CPictureStatic::SetPicture(HBITMAP hBitmap)\n{\n\tm_memDC.DeleteDC();\n\tm_bitmap.Detach();\n\tif (!m_bitmap.Attach(hBitmap))\n\t\treturn;\n\t//ȡͼʵʴС\n\tGetObject(m_bitmap, sizeof(BITMAP), &m_bm);\n\tCDC* pDC = GetDC();\n\tm_memDC.CreateCompatibleDC(pDC);\n\tm_memDC.SelectObject(&m_bitmap);\n\t//ȡؼС\n\tGetClientRect(m_rect);\n\t//ֶػ\n\tInvalidate();\n}\n\nBEGIN_MESSAGE_MAP(CPictureStatic, CStatic)\n\tON_WM_PAINT()\nEND_MESSAGE_MAP()\n\n\nvoid CPictureStatic::OnPaint()\n{\n\tCPaintDC dc(this); // device context for painting\n\t\t\t\t\t   // TODO: ڴ˴Ϣ\n\t\t\t\t\t   // ΪͼϢ CStatic::OnPaint()\n\tif (m_bitmap.m_hObject != NULL)\n\t{\n\t\t// бͼƬʧ\n\t\tdc.SetStretchBltMode(HALFTONE);\n\t\tdc.SetBrushOrg(0, 0);\n\t\t//ƽڴDCеͼ\n\t\tdc.StretchBlt(0, 0, m_rect.Width(), m_rect.Height(), &m_memDC, 0, 0, m_bm.bmWidth, m_bm.bmHeight, SRCCOPY);\n\n\t\t//򸸴ڷػϢ\n\t\tCWnd* pParent{ GetParent() };\n\t\tif (pParent != nullptr)\n\t\t\tpParent->SendMessage(WM_CONTROL_REPAINT, (WPARAM)this, LPARAM(&dc));\n\t}\n}\n"
  },
  {
    "path": "TrafficMonitor/PictureStatic.h",
    "content": "/*һʾͼƬľ̬ؼ\nҪʱSetPicture()þ̬ͼƬ\nؼػʱ򸸴ڷWM_CONTROL_REPAINTϢ\nͨwParamݵǰؼCWndָ룬ͨlParamCDCָ\n*/\n#pragma once\n#define WM_CONTROL_REPAINT (WM_USER + 1003)\t\t//ؼػϢ\n\nclass CPictureStatic : public CStatic\n{\npublic:\n\tCPictureStatic();\n\t~CPictureStatic();\n\n\tvoid SetPicture(UINT pic_id);\n\tvoid SetPicture(HBITMAP hBitmap);\n\nprotected:\n\tCDC m_memDC;\n\tCBitmap m_bitmap;\n\tCRect m_rect;\n\tBITMAP m_bm;\n\npublic:\n\tDECLARE_MESSAGE_MAP()\n\tafx_msg void OnPaint();\n};\n\n"
  },
  {
    "path": "TrafficMonitor/PluginInfoDlg.cpp",
    "content": "﻿// PluginInfoDlg.cpp: 实现文件\n//\n\n#include \"stdafx.h\"\n#include \"TrafficMonitor.h\"\n#include \"PluginInfoDlg.h\"\n#include \"FilePathHelper.h\"\n\n\n// CPluginInfoDlg 对话框\n\nIMPLEMENT_DYNAMIC(CPluginInfoDlg, CBaseDialog)\n\nCPluginInfoDlg::CPluginInfoDlg(int plugin_index, CWnd* pParent /*=nullptr*/)\n\t: CBaseDialog(IDD_NETWORK_INFO_DIALOG, pParent), m_cur_index(plugin_index)\n{\n    if (m_cur_index < 0 || m_cur_index >= static_cast<int>(theApp.m_plugins.GetPlugins().size()))\n        m_cur_index = 0;\n}\n\nCPluginInfoDlg::~CPluginInfoDlg()\n{\n}\n\nvoid CPluginInfoDlg::DoDataExchange(CDataExchange* pDX)\n{\n\tCBaseDialog::DoDataExchange(pDX);\n    DDX_Control(pDX, IDC_INFO_LIST1, m_info_list);\n}\n\nvoid CPluginInfoDlg::ShowInfo()\n{\n    if (m_cur_index >= 0 && m_cur_index < static_cast<int>(theApp.m_plugins.GetPlugins().size()))\n    {\n        auto& plugin_info = theApp.m_plugins.GetPlugins()[m_cur_index];\n        m_info_list.SetItemText(RI_NAME, 1, plugin_info.Property(ITMPlugin::TMI_NAME).c_str());\n        m_info_list.SetItemText(RI_DESCRIPTION, 1, plugin_info.Property(ITMPlugin::TMI_DESCRIPTION).c_str());\n        m_info_list.SetItemText(RI_FILE_NAME, 1, CFilePathHelper(plugin_info.file_path).GetFileName().c_str());\n        m_info_list.SetItemText(RI_FILE_PATH, 1, plugin_info.file_path.c_str());\n        m_info_list.SetItemText(RI_ITEM_NUM, 1, std::to_wstring(plugin_info.plugin_items.size()).c_str());\n        wstring item_names;\n        wstring item_id;\n        for (const auto& item : plugin_info.plugin_items)\n        {\n            item_names += item->GetItemName();\n            item_names += L\";\";\n            item_id += item->GetItemId();\n            item_id += L\";\";\n        }\n        if (!plugin_info.plugin_items.empty())\n        {\n            item_names.pop_back();\n            item_id.pop_back();\n        }\n        m_info_list.SetItemText(RI_ITEM_NAMES, 1, item_names.c_str());\n        m_info_list.SetItemText(RI_ITEM_ID, 1, item_id.c_str());\n        m_info_list.SetItemText(RI_AUTHOR, 1, plugin_info.Property(ITMPlugin::TMI_AUTHOR).c_str());\n        m_info_list.SetItemText(RI_COPYRIGHT, 1, plugin_info.Property(ITMPlugin::TMI_COPYRIGHT).c_str());\n        m_info_list.SetItemText(RI_URL, 1, plugin_info.Property(ITMPlugin::TMI_URL).c_str());\n        m_info_list.SetItemText(RI_VERSION, 1, plugin_info.Property(ITMPlugin::TMI_VERSION).c_str());\n        if (plugin_info.plugin != nullptr)\n            m_info_list.SetItemText(RI_API_VERSION, 1, std::to_wstring(plugin_info.plugin->GetAPIVersion()).c_str());\n    }\n\n    //显示当前选择指示\n    CString str;\n    str.Format(_T(\"%d/%d\"), m_cur_index + 1, theApp.m_plugins.GetPlugins().size());\n    SetDlgItemText(IDC_INDEX_STATIC, str);\n}\n\n\nBEGIN_MESSAGE_MAP(CPluginInfoDlg, CBaseDialog)\n    ON_COMMAND(ID_COPY_TEXT, &CPluginInfoDlg::OnCopyText)\n    ON_NOTIFY(NM_RCLICK, IDC_INFO_LIST1, &CPluginInfoDlg::OnNMRClickInfoList1)\n    ON_BN_CLICKED(IDC_PREVIOUS_BUTTON, &CPluginInfoDlg::OnBnClickedPreviousButton)\n    ON_BN_CLICKED(IDC_NEXT_BUTTON, &CPluginInfoDlg::OnBnClickedNextButton)\n    ON_WM_MOUSEWHEEL()\nEND_MESSAGE_MAP()\n\n\n// CPluginInfoDlg 消息处理程序\n\n\nBOOL CPluginInfoDlg::OnInitDialog()\n{\n    CBaseDialog::OnInitDialog();\n\n    // TODO:  在此添加额外的初始化\n    SetWindowText(CCommon::LoadText(IDS_PLUGIN_INFO));\n    SetIcon(theApp.GetMenuIcon(IDI_PLUGINS), FALSE);\t\t// 设置小图标\n\n    //初始化列表控件\n    CRect rect;\n    m_info_list.GetClientRect(rect);\n    m_info_list.SetExtendedStyle(LVS_EX_FULLROWSELECT | LVS_EX_LABELTIP);\n    int width0, width1;\n    width0 = rect.Width() / 4;\n    width1 = rect.Width() - width0 - theApp.DPI(20) - 1;\n    m_info_list.InsertColumn(0, CCommon::LoadText(IDS_ITEM), LVCFMT_LEFT, width0);\n    m_info_list.InsertColumn(1, CCommon::LoadText(IDS_VALUE), LVCFMT_LEFT, width1);\n\n    //向列表中插入行\n    for (int i = 0; i < RI_MAX; i++)\n    {\n        m_info_list.InsertItem(i, GetRowName(i));\n    }\n\n    //显示列表中的信息\n    ShowInfo();\n\n    m_info_list.GetToolTips()->SetWindowPos(&wndTopMost, 0, 0, 0, 0, SWP_NOSIZE | SWP_NOMOVE);\n\n    m_menu.LoadMenu(IDR_INFO_MENU); //装载右键菜单\n\n    return TRUE;  // return TRUE unless you set the focus to a control\n                  // 异常: OCX 属性页应返回 FALSE\n}\n\nCString CPluginInfoDlg::GetDialogName() const\n{\n    return _T(\"PluginInfoDlg\");\n}\n\nvoid CPluginInfoDlg::OnCopyText()\n{\n    if (!CCommon::CopyStringToClipboard(wstring(m_selected_string)))\n        MessageBox(CCommon::LoadText(IDS_COPY_TO_CLIPBOARD_FAILED), NULL, MB_ICONWARNING);\n}\n\nvoid CPluginInfoDlg::OnNMRClickInfoList1(NMHDR* pNMHDR, LRESULT* pResult)\n{\n    LPNMITEMACTIVATE pNMItemActivate = reinterpret_cast<LPNMITEMACTIVATE>(pNMHDR);\n    // TODO: 在此添加控件通知处理程序代码\n\n    //获取鼠标点击处的文本\n    int item, sub_item;\n    item = pNMItemActivate->iItem;\n    sub_item = pNMItemActivate->iSubItem;\n    m_selected_string = m_info_list.GetItemText(item, sub_item);\n\n    //弹出右键菜单\n    CMenu* pContextMenu = m_menu.GetSubMenu(0);\t//获取第一个弹出菜单\n    CPoint point1;\t//定义一个用于确定光标位置的位置\n    GetCursorPos(&point1);\t//获取当前光标的位置，以便使得菜单可以跟随光标\n    pContextMenu->TrackPopupMenu(TPM_LEFTALIGN | TPM_RIGHTBUTTON, point1.x, point1.y, this); //在指定位置显示弹出菜单\n\n    *pResult = 0;\n}\n\nvoid CPluginInfoDlg::OnBnClickedPreviousButton()\n{\n    if (m_cur_index > 0)\n    {\n        m_cur_index--;\n        ShowInfo();\n    }\n}\n\nvoid CPluginInfoDlg::OnBnClickedNextButton()\n{\n    if (m_cur_index < static_cast<int>(theApp.m_plugins.GetPlugins().size() - 1))\n    {\n        m_cur_index++;\n        ShowInfo();\n    }\n}\n\nBOOL CPluginInfoDlg::PreTranslateMessage(MSG* pMsg)\n{\n    if (pMsg->message == WM_KEYDOWN)\n    {\n        if (pMsg->wParam == VK_LEFT)\n        {\n            OnBnClickedPreviousButton();\n            return TRUE;\n        }\n        if (pMsg->wParam == VK_RIGHT)\n        {\n            OnBnClickedNextButton();\n            return TRUE;\n        }\n    }\n\n    return CBaseDialog::PreTranslateMessage(pMsg);\n}\n\n\nBOOL CPluginInfoDlg::OnMouseWheel(UINT nFlags, short zDelta, CPoint pt)\n{\n    //通过鼠标滚轮翻页\n    if (zDelta > 0)\n    {\n        OnBnClickedPreviousButton();\n    }\n    if (zDelta < 0)\n    {\n        OnBnClickedNextButton();\n    }\n\n    return CBaseDialog::OnMouseWheel(nFlags, zDelta, pt);\n}\n\nCString CPluginInfoDlg::GetRowName(int row_index)\n{\n    switch (row_index)\n    {\n    case CPluginInfoDlg::RI_NAME:\n        return CCommon::LoadText(IDS_NAME);\n    case CPluginInfoDlg::RI_DESCRIPTION:\n        return CCommon::LoadText(IDS_DESCRIPTION);\n    case CPluginInfoDlg::RI_FILE_NAME:\n        return CCommon::LoadText(IDS_FILE_NAME);\n    case CPluginInfoDlg::RI_FILE_PATH:\n        return CCommon::LoadText(IDS_FILE_PATH);\n    case CPluginInfoDlg::RI_ITEM_NUM:\n        return CCommon::LoadText(IDS_ITEM_NUM);\n    case CPluginInfoDlg::RI_ITEM_NAMES:\n        return CCommon::LoadText(IDS_ITEM_NAMES);\n    case CPluginInfoDlg::RI_ITEM_ID:\n        return CCommon::LoadText(IDS_DISP_ITEM_ID);\n    case CPluginInfoDlg::RI_AUTHOR:\n        return CCommon::LoadText(IDS_AUTHOR);\n    case CPluginInfoDlg::RI_COPYRIGHT:\n        return CCommon::LoadText(IDS_COPYRIGHT);\n    case CPluginInfoDlg::RI_URL:\n        return CCommon::LoadText(IDS_URL);\n    case CPluginInfoDlg::RI_VERSION:\n        return CCommon::LoadText(IDS_VERSION);\n    case CPluginInfoDlg::RI_API_VERSION:\n        return CCommon::LoadText(IDS_PLUGIN_API_VERSION);\n    default:\n        break;\n    }\n    return CString();\n}\n"
  },
  {
    "path": "TrafficMonitor/PluginInfoDlg.h",
    "content": "﻿#pragma once\n#include \"BaseDialog.h\"\n\n// CPluginInfoDlg 对话框\n\nclass CPluginInfoDlg : public CBaseDialog\n{\n\tDECLARE_DYNAMIC(CPluginInfoDlg)\n\npublic:\n\tCPluginInfoDlg(int plugin_index, CWnd* pParent = nullptr);   // 标准构造函数\n\tvirtual ~CPluginInfoDlg();\n\n// 对话框数据\n#ifdef AFX_DESIGN_TIME\n\tenum { IDD = IDD_NETWORK_INFO_DIALOG };\n#endif\n\nprivate:\n    int m_cur_index;        //初始显示的插件索引\n    CListCtrl m_info_list;\n    CMenu m_menu;\n    CString m_selected_string;\n\n    //列表中的列\n    enum RowIndex\n    {\n        RI_FILE_NAME,\n        RI_FILE_PATH,\n        RI_NAME,\n        RI_DESCRIPTION,\n        RI_ITEM_NUM,\n        RI_ITEM_NAMES,\n        RI_ITEM_ID,\n        RI_AUTHOR,\n        RI_COPYRIGHT,\n        RI_URL,\n        RI_VERSION,\n        RI_API_VERSION,\n        RI_MAX\n    };\n\nprotected:\n\tvirtual void DoDataExchange(CDataExchange* pDX);    // DDX/DDV 支持\n    void ShowInfo();\n    static CString GetRowName(int row_index);   //获取行的名称\n\n    // 通过 CBaseDialog 继承\n    virtual CString GetDialogName() const override;\n\n\tDECLARE_MESSAGE_MAP()\npublic:\n    virtual BOOL OnInitDialog();\n\n    afx_msg void OnCopyText();\n    afx_msg void OnNMRClickInfoList1(NMHDR* pNMHDR, LRESULT* pResult);\n    afx_msg void OnBnClickedPreviousButton();\n    afx_msg void OnBnClickedNextButton();\n    virtual BOOL PreTranslateMessage(MSG* pMsg);\n    afx_msg BOOL OnMouseWheel(UINT nFlags, short zDelta, CPoint pt);\n\n};\n"
  },
  {
    "path": "TrafficMonitor/PluginManager.cpp",
    "content": "﻿#include \"stdafx.h\"\n#include \"PluginManager.h\"\n#include \"Common.h\"\n#include \"TrafficMonitor.h\"\n\n#define PLUGIN_UNSUPPORT_VERSION 0      //不被支持的插件版本\n\nCPluginManager::CPluginManager()\n{\n}\n\nCPluginManager::~CPluginManager()\n{\n    //卸载插件\n    for (const auto& m : m_modules)\n        FreeLibrary(m.plugin_module);\n}\n\nstatic wstring WcharArrayToWString(const wchar_t* str)\n{\n    if (str == nullptr)\n        return wstring();\n    else\n        return wstring(str);\n}\n\nvoid CPluginManager::LoadPlugins()\n{\n    //从plugins目录下加载插件\n    wstring plugin_dir;\n    plugin_dir = CCommon::GetModuleDir() + L\"plugins\";\n    vector<wstring> plugin_files;\n    CCommon::GetFiles((plugin_dir + L\"\\\\*.dll\").c_str(), plugin_files);\t\t//获取Plugins目录下所有的dll文件的文件名\n    for (const auto& file : plugin_files)\n    {\n        //插件信息\n        m_modules.push_back(PluginInfo());\n        PluginInfo& plugin_info{ m_modules.back() };\n        //插件dll的路径\n        plugin_info.file_path = plugin_dir + file;\n        //插件文件名\n        std::wstring file_name{ file };\n        if (!file_name.empty() && (file_name[0] == L'\\\\' || file_name[0] == L'/'))\n            file_name = file_name.substr(1);\n        //如果插件被禁用，则不加载插件\n        if (theApp.m_cfg_data.plugin_disabled.Contains(file_name))\n        {\n            plugin_info.state = PluginState::PS_DISABLE;\n            continue;\n        }\n        //载入dll\n        plugin_info.plugin_module = LoadLibrary(plugin_info.file_path.c_str());\n        if (plugin_info.plugin_module == NULL)\n        {\n            plugin_info.state = PluginState::PS_MUDULE_LOAD_FAILED;\n            plugin_info.error_code = GetLastError();\n            continue;\n        }\n        //获取函数的入口地址\n        pfTMPluginGetInstance TMPluginGetInstance = (pfTMPluginGetInstance)::GetProcAddress(plugin_info.plugin_module, \"TMPluginGetInstance\");\n        if (TMPluginGetInstance == NULL)\n        {\n            plugin_info.state = PluginState::PS_FUNCTION_GET_FAILED;\n            plugin_info.error_code = GetLastError();\n            continue;\n        }\n        //创建插件对象\n        plugin_info.plugin = TMPluginGetInstance();\n        if (plugin_info.plugin == nullptr)\n            continue;\n        //检查插件版本\n        int version = plugin_info.plugin->GetAPIVersion();\n        if (version <= PLUGIN_UNSUPPORT_VERSION)\n        {\n            plugin_info.state = PluginState::PS_VERSION_NOT_SUPPORT;\n            continue;\n        }\n        //获取插件信息\n        for (int i{}; i < ITMPlugin::TMI_MAX; i++)\n        {\n            ITMPlugin::PluginInfoIndex index{ static_cast<ITMPlugin::PluginInfoIndex>(i) };\n            plugin_info.properties[index] = WcharArrayToWString(plugin_info.plugin->GetInfo(index));\n        }\n\n        //获取插件显示项目\n        int index = 0;\n        while (true)\n        {\n            IPluginItem* item = plugin_info.plugin->GetItem(index);\n            if (item == nullptr)\n                break;\n            plugin_info.plugin_items.push_back(item);\n            m_plugins.push_back(item);\n            m_plguin_item_map[item] = plugin_info.plugin;\n            index++;\n        }\n    }\n\n    //初始化所有任务栏显示项目\n    for (const auto& display_item : AllDisplayItems)\n    {\n        m_all_display_items_with_plugins.insert(display_item);\n    }\n    for (const auto& display_item : m_plugins)\n    {\n        m_all_display_items_with_plugins.insert(display_item);\n    }\n}\n\nconst std::vector<IPluginItem*>& CPluginManager::GetPluginItems() const\n{\n    return m_plugins;\n}\n\nconst std::vector<CPluginManager::PluginInfo>& CPluginManager::GetPlugins() const\n{\n    return m_modules;\n}\n\nIPluginItem* CPluginManager::GetItemById(const std::wstring& item_id)\n{\n    for (const auto& item : m_plugins)\n    {\n        if (item->GetItemId() == item_id)\n            return item;\n    }\n    return nullptr;\n}\n\nIPluginItem* CPluginManager::GetItemByIndex(int index)\n{\n    if (index >= 0 && index < static_cast<int>(m_plugins.size()))\n        return m_plugins[index];\n    return nullptr;\n}\n\nint CPluginManager::GetItemIndex(IPluginItem* item) const\n{\n    for (auto iter = m_plugins.begin(); iter != m_plugins.end(); ++iter)\n    {\n        if (*iter == item)\n            return iter - m_plugins.begin();\n    }\n    return -1;\n}\n\nITMPlugin* CPluginManager::GetPluginByItem(IPluginItem* pItem)\n{\n    if (pItem == nullptr)\n        return nullptr;\n    return m_plguin_item_map[pItem];\n}\n\nvoid CPluginManager::EnumPlugin(std::function<void(ITMPlugin*)> func) const\n{\n    for (const auto& item : m_modules)\n    {\n        if (item.plugin != nullptr)\n        {\n            func(item.plugin);\n        }\n    }\n}\n\nvoid CPluginManager::EnumPluginItem(std::function<void(IPluginItem*)> func) const\n{\n    for (const auto& item : m_plugins)\n    {\n        if (item != nullptr)\n        {\n            func(item);\n        }\n    }\n}\n\nconst std::set<CommonDisplayItem>& CPluginManager::AllDisplayItemsWithPlugins()\n{\n    return m_all_display_items_with_plugins;\n}\n\n\nint CPluginManager::GetItemWidth(IPluginItem* pItem, CDC* pDC)\n{\n    int width = 0;\n    ITMPlugin* plugin = GetPluginByItem(pItem);\n    if (plugin != nullptr && plugin->GetAPIVersion() >= 3)\n    {\n        width = pItem->GetItemWidthEx(pDC->GetSafeHdc());       //优先使用GetItemWidthEx接口获取宽度\n    }\n    if (width == 0)\n    {\n        width = theApp.DPI(pItem->GetItemWidth());\n    }\n    return width;\n}\n\nstd::wstring CPluginManager::PluginInfo::Property(ITMPlugin::PluginInfoIndex index) const\n{\n    auto iter = properties.find(index);\n    if (iter != properties.end())\n        return iter->second;\n    return wstring();\n}\n"
  },
  {
    "path": "TrafficMonitor/PluginManager.h",
    "content": "﻿#pragma once\n#include \"PluginInterface.h\"\n#include <map>\n#include \"TaskbarItemOrderHelper.h\"\n#include <functional>\n\ntypedef ITMPlugin* (*pfTMPluginGetInstance)();\n\n//用于加载和管理插件\nclass CPluginManager\n{\npublic:\n    //插件的状态\n    enum class PluginState\n    {\n        PS_SUCCEED,             //载入成功\n        PS_MUDULE_LOAD_FAILED,  //dll加载失败\n        PS_FUNCTION_GET_FAILED, //插件函数获取失败\n        PS_VERSION_NOT_SUPPORT, //插件版本不被支持\n        PS_DISABLE              //已禁用\n    };\n\n    //插件信息\n    struct PluginInfo\n    {\n        wstring file_path;      //文件路径\n        HMODULE plugin_module{};  //dll module\n        ITMPlugin* plugin{};      //插件对象\n        std::vector<IPluginItem*> plugin_items; //插件提供的所有显示项目\n        PluginState state{};    //插件的状态\n        DWORD error_code{};     //错误代码（GetLastError的返回值）\n        std::map<ITMPlugin::PluginInfoIndex, std::wstring> properties;    //插件属性\n        std::wstring Property(ITMPlugin::PluginInfoIndex) const;\n    };\n\n    CPluginManager();\n    ~CPluginManager();\n    void LoadPlugins();\n\n    const std::vector<IPluginItem*>& GetPluginItems() const;\n    const std::vector<PluginInfo>& GetPlugins() const;\n    IPluginItem* GetItemById(const std::wstring& item_id);\n    IPluginItem* GetItemByIndex(int index);\n    int GetItemIndex(IPluginItem* item) const;\n    ITMPlugin* GetPluginByItem(IPluginItem* pItem);\n\n    //遍历所有插件\n    //func: 参数为遍历到的ITMPlugin对象\n    void EnumPlugin(std::function<void(ITMPlugin*)> func) const;\n\n    //遍历所有插件项目\n    //func: 参数为遍历到的IPluginItem对象\n    void EnumPluginItem(std::function<void(IPluginItem*)> func) const;\n\n    const std::set<CommonDisplayItem>& AllDisplayItemsWithPlugins();\n\n    int GetItemWidth(IPluginItem* pItem, CDC* pDC);\n\nprivate:\n    std::vector<IPluginItem*> m_plugins;\n    std::vector<PluginInfo> m_modules;\n    std::set<CommonDisplayItem> m_all_display_items_with_plugins;   //包含插件在内的所有任务栏显示项目\n    std::map<IPluginItem*, ITMPlugin*> m_plguin_item_map;          //用于根据插件项目查找对应插件的map\n};\n"
  },
  {
    "path": "TrafficMonitor/PluginManagerDlg.cpp",
    "content": "﻿// PluginManagerDlg.cpp: 实现文件\n//\n\n#include \"stdafx.h\"\n#include \"TrafficMonitor.h\"\n#include \"PluginManagerDlg.h\"\n#include \"FilePathHelper.h\"\n#include \"PluginInfoDlg.h\"\n\n\n// CPluginManagerDlg 对话框\n\nIMPLEMENT_DYNAMIC(CPluginManagerDlg, CBaseDialog)\n\nCPluginManagerDlg::CPluginManagerDlg(CWnd* pParent /*=nullptr*/)\n    : CBaseDialog(IDD_PLUGIN_MANAGER_DIALOG, pParent)\n{\n\n}\n\nCPluginManagerDlg::~CPluginManagerDlg()\n{\n}\n\nvoid CPluginManagerDlg::DoDataExchange(CDataExchange* pDX)\n{\n    CBaseDialog::DoDataExchange(pDX);\n    DDX_Control(pDX, IDC_LIST1, m_list_ctrl);\n    DDX_Control(pDX, IDC_PLUGIN_DOWNLOAD_STATIC, m_plugin_download_lnk);\n    DDX_Control(pDX, IDC_PLUGIN_DEV_GUID_STATIC, m_plugin_dev_guide_lnk);\n    DDX_Control(pDX, IDC_OPEN_PLUGIN_DIR_STATIC, m_open_plugin_dir_lnk);\n}\n\nvoid CPluginManagerDlg::EnableControl()\n{\n    bool enable{ IsSelectedPluginEnable() };\n    EnableDlgCtrl(IDC_OPTINS_BUTTON, enable);\n    EnableDlgCtrl(IDC_PLUGIN_INFO_BUTTON, enable);\n}\n\nbool CPluginManagerDlg::IsSelectedValid()\n{\n    return m_item_selected >= 0 && m_item_selected < static_cast<int>(theApp.m_plugins.GetPlugins().size());\n}\n\nbool CPluginManagerDlg::IsSelectedPluginEnable()\n{\n    CPluginManager::PluginInfo plugin_info{};\n    bool plugin_loaded{ false };\n    if (IsSelectedValid())\n    {\n        plugin_info = theApp.m_plugins.GetPlugins()[m_item_selected];\n        plugin_loaded = (plugin_info.state == CPluginManager::PluginState::PS_SUCCEED);\n    }\n\n    return plugin_loaded;\n}\n\nCString CPluginManagerDlg::GetDialogName() const\n{\n    return _T(\"PluginManagerDlg\");\n}\n\n\nBEGIN_MESSAGE_MAP(CPluginManagerDlg, CBaseDialog)\n    ON_NOTIFY(NM_RCLICK, IDC_LIST1, &CPluginManagerDlg::OnNMRClickList1)\n    ON_NOTIFY(NM_CLICK, IDC_LIST1, &CPluginManagerDlg::OnNMClickList1)\n    ON_BN_CLICKED(IDC_OPTINS_BUTTON, &CPluginManagerDlg::OnBnClickedOptinsButton)\n    ON_BN_CLICKED(IDC_PLUGIN_INFO_BUTTON, &CPluginManagerDlg::OnBnClickedPluginInfoButton)\n    ON_NOTIFY(NM_DBLCLK, IDC_LIST1, &CPluginManagerDlg::OnNMDblclkList1)\n    ON_WM_INITMENU()\n    ON_COMMAND(ID_PLUGIN_DETAIL, &CPluginManagerDlg::OnPluginDetail)\n    ON_COMMAND(ID_PLUGIN_OPTIONS, &CPluginManagerDlg::OnPluginOptions)\n    ON_COMMAND(ID_PLUGIN_DISABLE, &CPluginManagerDlg::OnPluginDisable)\n    ON_MESSAGE(WM_LINK_CLICKED, &CPluginManagerDlg::OnLinkClicked)\nEND_MESSAGE_MAP()\n\n\n// CPluginManagerDlg 消息处理程序\n\n\nBOOL CPluginManagerDlg::OnInitDialog()\n{\n    CBaseDialog::OnInitDialog();\n\n    // TODO:  在此添加额外的初始化\n    SetIcon(theApp.GetMenuIcon(IDI_PLUGINS), FALSE);\n\n    //初始化列表控件\n    CRect rect;\n    m_list_ctrl.GetClientRect(rect);\n    m_list_ctrl.SetExtendedStyle(LVS_EX_FULLROWSELECT | LVS_EX_LABELTIP);\n    int width0, width1, width2;\n    width0 = width1 = rect.Width() / 3;\n    width2 = rect.Width() - width0 - width1 - theApp.DPI(20) - 1;\n    m_list_ctrl.InsertColumn(0, CCommon::LoadText(IDS_FILE_NAME), LVCFMT_LEFT, width0);\n    m_list_ctrl.InsertColumn(1, CCommon::LoadText(IDS_PLUGIN_NAME), LVCFMT_LEFT, width1);\n    m_list_ctrl.InsertColumn(2, CCommon::LoadText(IDS_STATUS), LVCFMT_LEFT, width2);\n\n    //向列表中插入行\n    for (const auto& plugin : theApp.m_plugins.GetPlugins())\n    {\n        std::wstring file_name = CFilePathHelper(plugin.file_path).GetFileName();\n        CString status;\n        switch (plugin.state)\n        {\n        case CPluginManager::PluginState::PS_SUCCEED:\n            status = CCommon::LoadText(IDS_PLUGIN_LOAD_SUCCEED);\n            break;\n        case CPluginManager::PluginState::PS_MUDULE_LOAD_FAILED:\n            status = CCommon::LoadTextFormat(IDS_PLUGIN_MODULE_LOAD_FAILED, { static_cast<int>(plugin.error_code) });\n            break;\n        case CPluginManager::PluginState::PS_FUNCTION_GET_FAILED:\n            status = CCommon::LoadTextFormat(IDS_PLUGIN_FUNCTION_GET_FAILED, { static_cast<int>(plugin.error_code) });\n            break;\n        case CPluginManager::PluginState::PS_DISABLE:\n            status = CCommon::LoadText(IDS_DISABLED);\n            break;\n        case CPluginManager::PluginState::PS_VERSION_NOT_SUPPORT:\n            status = CCommon::LoadText(IDS_PLUGIN_VERSION_NOT_SUPPORT);\n            break;\n        default:\n            break;\n        }\n        int index = m_list_ctrl.GetItemCount();\n        m_list_ctrl.InsertItem(index, file_name.c_str());\n        m_list_ctrl.SetItemText(index, 1, plugin.Property(ITMPlugin::TMI_NAME).c_str());\n        m_list_ctrl.SetItemText(index, 2, status);\n    }\n\n    m_plugin_download_lnk.SetURL(L\"https://github.com/zhongyang219/TrafficMonitorPlugins/blob/main/download/plugin_download.md\");\n    m_plugin_dev_guide_lnk.SetURL(L\"https://github.com/zhongyang219/TrafficMonitor/wiki/%E6%8F%92%E4%BB%B6%E5%BC%80%E5%8F%91%E6%8C%87%E5%8D%97\");\n    m_open_plugin_dir_lnk.SetLinkIsURL(false);\n\n    EnableControl();\n\n    m_menu.LoadMenu(IDR_PLUGIN_MANAGER_MENU); //装载右键菜单\n\n    return TRUE;  // return TRUE unless you set the focus to a control\n                  // 异常: OCX 属性页应返回 FALSE\n}\n\n\nvoid CPluginManagerDlg::OnNMRClickList1(NMHDR* pNMHDR, LRESULT* pResult)\n{\n    LPNMITEMACTIVATE pNMItemActivate = reinterpret_cast<LPNMITEMACTIVATE>(pNMHDR);\n    // TODO: 在此添加控件通知处理程序代码\n    m_item_selected = pNMItemActivate->iItem;\n    EnableControl();\n\n    //弹出右键菜单\n    CMenu* pContextMenu = m_menu.GetSubMenu(0);\t//获取第一个弹出菜单\n    CPoint point1;\t//定义一个用于确定光标位置的位置\n    GetCursorPos(&point1);\t//获取当前光标的位置，以便使得菜单可以跟随光标\n    pContextMenu->TrackPopupMenu(TPM_LEFTALIGN | TPM_RIGHTBUTTON, point1.x, point1.y, this); //在指定位置显示弹出菜单\n\n    *pResult = 0;\n}\n\n\nvoid CPluginManagerDlg::OnNMClickList1(NMHDR* pNMHDR, LRESULT* pResult)\n{\n    LPNMITEMACTIVATE pNMItemActivate = reinterpret_cast<LPNMITEMACTIVATE>(pNMHDR);\n    // TODO: 在此添加控件通知处理程序代码\n    m_item_selected = pNMItemActivate->iItem;\n    EnableControl();\n    *pResult = 0;\n}\n\n\nvoid CPluginManagerDlg::OnBnClickedOptinsButton()\n{\n    // TODO: 在此添加控件通知处理程序代码\n    if (m_item_selected >= 0 && m_item_selected < static_cast<int>(theApp.m_plugins.GetPlugins().size()))\n    {\n        auto plugin_info = theApp.m_plugins.GetPlugins()[m_item_selected];\n        if (plugin_info.plugin != nullptr)\n        {\n            ITMPlugin::OptionReturn rtn = plugin_info.plugin->ShowOptionsDialog(m_hWnd);\n            if (rtn == ITMPlugin::OR_OPTION_NOT_PROVIDED)\n                MessageBox(CCommon::LoadText(IDS_PLUGIN_NO_OPTIONS_INFO), nullptr, MB_ICONINFORMATION | MB_OK);\n            //else if (rtn == ITMPlugin::OR_OPTION_CHANGED)\n            //    theApp.m_pMainWnd->SendMessage(WM_REOPEN_TASKBAR_WND);\n        }\n    }\n}\n\n\nvoid CPluginManagerDlg::OnBnClickedPluginInfoButton()\n{\n    // TODO: 在此添加控件通知处理程序代码\n    if (IsSelectedPluginEnable())\n    {\n        CPluginInfoDlg dlg(m_item_selected);\n        dlg.DoModal();\n    }\n}\n\n\nvoid CPluginManagerDlg::OnNMDblclkList1(NMHDR* pNMHDR, LRESULT* pResult)\n{\n    LPNMITEMACTIVATE pNMItemActivate = reinterpret_cast<LPNMITEMACTIVATE>(pNMHDR);\n    // TODO: 在此添加控件通知处理程序代码\n    m_item_selected = pNMItemActivate->iItem;\n    OnBnClickedPluginInfoButton();\n    *pResult = 0;\n}\n\n\nvoid CPluginManagerDlg::OnInitMenu(CMenu* pMenu)\n{\n    CBaseDialog::OnInitMenu(pMenu);\n\n    // TODO: 在此处添加消息处理程序代码\n    bool enable{ IsSelectedPluginEnable() };\n    pMenu->EnableMenuItem(ID_PLUGIN_DETAIL, MF_BYCOMMAND | (enable ? MF_ENABLED : MF_GRAYED));\n    pMenu->EnableMenuItem(ID_PLUGIN_OPTIONS, MF_BYCOMMAND | (enable ? MF_ENABLED : MF_GRAYED));\n    pMenu->EnableMenuItem(ID_PLUGIN_DISABLE, MF_BYCOMMAND | (IsSelectedValid() ? MF_ENABLED : MF_GRAYED));\n\n    bool disabled{};\n    CPluginManager::PluginInfo plugin_info;\n    if (m_item_selected >= 0 && m_item_selected < static_cast<int>(theApp.m_plugins.GetPlugins().size()))\n    {\n        plugin_info = theApp.m_plugins.GetPlugins()[m_item_selected];\n        std::wstring file_name = CFilePathHelper(plugin_info.file_path).GetFileName();\n        disabled = theApp.m_cfg_data.plugin_disabled.Contains(file_name);\n    }\n    pMenu->CheckMenuItem(ID_PLUGIN_DISABLE, MF_BYCOMMAND | (disabled ? MF_CHECKED : MF_UNCHECKED));\n}\n\n\nvoid CPluginManagerDlg::OnPluginDetail()\n{\n    OnBnClickedOptinsButton();\n}\n\n\nvoid CPluginManagerDlg::OnPluginOptions()\n{\n    OnBnClickedOptinsButton();\n}\n\n\nvoid CPluginManagerDlg::OnPluginDisable()\n{\n    if (m_item_selected >= 0 && m_item_selected < static_cast<int>(theApp.m_plugins.GetPlugins().size()))\n    {\n        CPluginManager::PluginInfo plugin_info = theApp.m_plugins.GetPlugins()[m_item_selected];\n        std::wstring file_name = CFilePathHelper(plugin_info.file_path).GetFileName();\n        bool disabled = theApp.m_cfg_data.plugin_disabled.Contains(file_name);\n        theApp.m_cfg_data.plugin_disabled.SetStrContained(file_name, !disabled);\n        MessageBox(CCommon::LoadText(IDS_RESTART_TO_APPLY_CHANGE_INFO), nullptr, MB_OK | MB_ICONINFORMATION);\n    }\n}\n\n\nafx_msg LRESULT CPluginManagerDlg::OnLinkClicked(WPARAM wParam, LPARAM lParam)\n{\n    CWnd* pCtrl = (CWnd*)wParam;\n    //点击了“打开插件目录”\n    if (pCtrl == &m_open_plugin_dir_lnk)\n    {\n        wstring plugin_dir = CCommon::GetModuleDir() + L\"plugins\";\n        CreateDirectory(plugin_dir.c_str(), NULL);       //如果plugins不存在，则创建它\n        ShellExecute(NULL, _T(\"open\"), _T(\"explorer\"), plugin_dir.c_str(), NULL, SW_SHOWNORMAL);\n    }\n\n    return 0;\n}\n"
  },
  {
    "path": "TrafficMonitor/PluginManagerDlg.h",
    "content": "﻿#pragma once\n#include \"BaseDialog.h\"\n#include \"LinkStatic.h\"\n\n// CPluginManagerDlg 对话框\n\nclass CPluginManagerDlg : public CBaseDialog\n{\n    DECLARE_DYNAMIC(CPluginManagerDlg)\n\npublic:\n    CPluginManagerDlg(CWnd* pParent = nullptr);   // 标准构造函数\n    virtual ~CPluginManagerDlg();\n\n    // 对话框数据\n#ifdef AFX_DESIGN_TIME\n    enum { IDD = IDD_PLUGIN_MANAGER_DIALOG };\n#endif\n\nprivate:\n    CListCtrl m_list_ctrl;\n    int m_item_selected{ -1 };\n    CMenu m_menu;\n    CLinkStatic m_plugin_download_lnk;\n    CLinkStatic m_plugin_dev_guide_lnk;\n    CLinkStatic m_open_plugin_dir_lnk;\n\nprotected:\n    virtual void DoDataExchange(CDataExchange* pDX);    // DDX/DDV 支持\n    void EnableControl();\n    bool IsSelectedValid();\n    bool IsSelectedPluginEnable();     //选中插件是否可用\n\n    DECLARE_MESSAGE_MAP()\n\n    // 通过 CBaseDialog 继承\n    virtual CString GetDialogName() const override;\npublic:\n    virtual BOOL OnInitDialog();\n    afx_msg void OnNMRClickList1(NMHDR* pNMHDR, LRESULT* pResult);\n    afx_msg void OnNMClickList1(NMHDR* pNMHDR, LRESULT* pResult);\n    afx_msg void OnBnClickedOptinsButton();\n    afx_msg void OnBnClickedPluginInfoButton();\n    afx_msg void OnNMDblclkList1(NMHDR* pNMHDR, LRESULT* pResult);\n    afx_msg void OnInitMenu(CMenu* pMenu);\n    afx_msg void OnPluginDetail();\n    afx_msg void OnPluginOptions();\n    afx_msg void OnPluginDisable();\nprotected:\n    afx_msg LRESULT OnLinkClicked(WPARAM wParam, LPARAM lParam);\n};\n"
  },
  {
    "path": "TrafficMonitor/ReadMe.txt",
    "content": "﻿================================================================================\n    MICROSOFT 基础类库 : TrafficMonitor 项目概述\n===============================================================================\n\n应用程序向导已为您创建了此 TrafficMonitor 应用程序。此应用程序不仅演示 Microsoft 基础类的基本使用方法，还可作为您编写应用程序的起点。\n\n本文件概要介绍组成 TrafficMonitor 应用程序的每个文件的内容。\n\nTrafficMonitor.vcxproj\n    这是使用应用程序向导生成的 VC++ 项目的主项目文件，其中包含生成该文件的 Visual C++ 的版本信息，以及有关使用应用程序向导选择的平台、配置和项目功能的信息。\n\nTrafficMonitor.vcxproj.filters\n    这是使用“应用程序向导”生成的 VC++ 项目筛选器文件。它包含有关项目文件与筛选器之间的关联信息。在 IDE 中，通过这种关联，在特定节点下以分组形式显示具有相似扩展名的文件。例如，“.cpp”文件与“源文件”筛选器关联。\n\nTrafficMonitor.h\n    这是应用程序的主头文件。\n    其中包括其他项目特定的标头（包括 Resource.h），并声明 CTrafficMonitorApp 应用程序类。\n\nTrafficMonitor.cpp\n    这是包含应用程序类 CTrafficMonitorApp 的主应用程序源文件。\n\nTrafficMonitor.rc\n    这是程序使用的所有 Microsoft Windows 资源的列表。它包括 RES 子目录中存储的图标、位图和光标。此文件可以直接在 Microsoft Visual C++ 中进行编辑。项目资源包含在 2052 中。\n\nres\\TrafficMonitor.ico\n    这是用作应用程序图标的图标文件。此图标包括在主资源文件 TrafficMonitor.rc 中。\n\nres\\TrafficMonitor.rc2\n    此文件包含不在 Microsoft Visual C++ 中进行编辑的资源。您应该将不可由资源编辑器编辑的所有资源放在此文件中。\n\n\n/////////////////////////////////////////////////////////////////////////////\n\n应用程序向导创建一个对话框类：\n\nTrafficMonitorDlg.h、TrafficMonitorDlg.cpp - 对话框\n    这些文件包含 CTrafficMonitorDlg 类。此类定义应用程序的主对话框的行为。对话框模板包含在 TrafficMonitor.rc 中，该文件可以在 Microsoft Visual C++ 中编辑。\n\n/////////////////////////////////////////////////////////////////////////////\n\n其他功能：\n\nActiveX 控件\n    该应用程序包含对使用 ActiveX 控件的支持。\n\n/////////////////////////////////////////////////////////////////////////////\n\n其他标准文件:\n\nStdAfx.h, StdAfx.cpp\n    这些文件用于生成名为 TrafficMonitor.pch 的预编译头 (PCH) 文件和名为 StdAfx.obj 的预编译类型文件。\n\nResource.h\n    这是标准头文件，可用于定义新的资源 ID。Microsoft Visual C++ 将读取并更新此文件。\n\nTrafficMonitor.manifest\n\tWindows XP 使用应用程序清单文件来描述特定版本的并行程序集的应用程序依赖项。加载程序使用这些信息来从程序集缓存中加载相应的程序集，并保护其不被应用程序访问。应用程序清单可能会包含在内，以作为与应用程序可执行文件安装在同一文件夹中的外部 .manifest 文件进行重新分发，它还可能以资源的形式包含在可执行文件中。\n/////////////////////////////////////////////////////////////////////////////\n\n其他注释:\n\n应用程序向导使用“TODO:”来指示应添加或自定义的源代码部分。\n\n如果应用程序使用共享 DLL 中的 MFC，您将需要重新分发 MFC DLL。如果应用程序所使用的语言与操作系统的区域设置不同，则还需要重新分发相应的本地化资源 mfc110XXX.DLL。\n有关上述话题的更多信息，请参见 MSDN 文档中有关重新分发 Visual C++ 应用程序的部分。\n\n/////////////////////////////////////////////////////////////////////////////\n"
  },
  {
    "path": "TrafficMonitor/SelectConnectionsDlg.cpp",
    "content": "﻿// SelectConnectionsDlg.cpp: 实现文件\n//\n\n#include \"stdafx.h\"\n#include \"TrafficMonitor.h\"\n#include \"SelectConnectionsDlg.h\"\n#include \"afxdialogex.h\"\n#include \"AdapterCommon.h\"\n\n// CSelectConnectionsDlg 对话框\n\nIMPLEMENT_DYNAMIC(CSelectConnectionsDlg, CBaseDialog)\n\nCSelectConnectionsDlg::CSelectConnectionsDlg(const StringSet& connections_hide, CWnd* pParent /*=nullptr*/)\n\t: m_connections_hide(connections_hide), CBaseDialog(IDD_SELECT_CONNECTIONS_DIALOG, pParent)\n{\n    std::vector<NetWorkConection> connections;\n    CAdapterCommon::GetAdapterInfo(connections);\n    for (const auto& item : connections)\n        m_all_connection_names.push_back(CCommon::StrToUnicode(item.description.c_str()));\n}\n\nCSelectConnectionsDlg::~CSelectConnectionsDlg()\n{\n}\n\nvoid CSelectConnectionsDlg::DoDataExchange(CDataExchange* pDX)\n{\n    CBaseDialog::DoDataExchange(pDX);\n    DDX_Control(pDX, IDC_LIST1, m_list_ctrl);\n}\n\n\nCString CSelectConnectionsDlg::GetDialogName() const\n{\n    //throw std::logic_error(\"The method or operation is not implemented.\");\n    return _T(\"SelectConnectionsDlg\");\n}\n\nBEGIN_MESSAGE_MAP(CSelectConnectionsDlg, CBaseDialog)\nEND_MESSAGE_MAP()\n\n\n// CSelectConnectionsDlg 消息处理程序\n\n\nBOOL CSelectConnectionsDlg::OnInitDialog()\n{\n    CBaseDialog::OnInitDialog();\n\n    // TODO:  在此添加额外的初始化\n    SetIcon(theApp.GetMenuIcon(IDI_CONNECTION), FALSE);\n    m_list_ctrl.SetItemHeight(0, theApp.DPI(20));\n\n    //向列表中添加项目\n    for (const auto& connection_name : m_all_connection_names)\n    {\n        m_list_ctrl.AddString(connection_name.c_str());\n        if (!m_connections_hide.Contains(connection_name))\n            m_list_ctrl.SetCheck(m_list_ctrl.GetCount() - 1, TRUE);\n        else\n            m_list_ctrl.SetCheck(m_list_ctrl.GetCount() - 1, FALSE);\n    }\n\n\n    return TRUE;  // return TRUE unless you set the focus to a control\n                  // 异常: OCX 属性页应返回 FALSE\n}\n\n\nvoid CSelectConnectionsDlg::OnOK()\n{\n    // TODO: 在此添加专用代码和/或调用基类\n    //保存每个项目的勾选状态\n    bool is_empty{ true };\n    for (int i = 0; i < static_cast<int>(m_all_connection_names.size()); i++)\n    {\n        bool is_checked = (m_list_ctrl.GetCheck(i) != 0);\n        if (is_checked)\n            is_empty = false;\n        m_connections_hide.SetStrContained(m_all_connection_names[i], !is_checked);\n    }\n\n    if (is_empty)\n    {\n        MessageBox(CCommon::LoadText(IDS_SELECT_AT_LEASE_ONE_WARNING), NULL, MB_ICONWARNING | MB_OK);\n        return;\n    }\n\n    CBaseDialog::OnOK();\n}\n"
  },
  {
    "path": "TrafficMonitor/SelectConnectionsDlg.h",
    "content": "﻿#pragma once\n#include \"CommonData.h\"\n#include \"BaseDialog.h\"\n\n// CSelectConnectionsDlg 对话框\n\nclass CSelectConnectionsDlg : public CBaseDialog\n{\n\tDECLARE_DYNAMIC(CSelectConnectionsDlg)\n\npublic:\n\tCSelectConnectionsDlg(const StringSet& connections_hide, CWnd* pParent = nullptr);   // 标准构造函数\n\tvirtual ~CSelectConnectionsDlg();\n\n    const StringSet& GetData() const { return m_connections_hide; }\n\n// 对话框数据\n#ifdef AFX_DESIGN_TIME\n\tenum { IDD = IDD_SELECT_CONNECTIONS_DIALOG };\n#endif\n\nprivate:\n    CCheckListBox m_list_ctrl;\n    std::vector<std::wstring> m_all_connection_names;\n    StringSet m_connections_hide;\n\nprotected:\n\tvirtual void DoDataExchange(CDataExchange* pDX);    // DDX/DDV 支持\n    virtual CString GetDialogName() const override;\n\n\tDECLARE_MESSAGE_MAP()\n\npublic:\n    virtual BOOL OnInitDialog();\n    virtual void OnOK();\n};\n"
  },
  {
    "path": "TrafficMonitor/SetItemOrderDlg.cpp",
    "content": "﻿// SetItemOrderDlg.cpp: 实现文件\n//\n\n#include \"stdafx.h\"\n#include \"TrafficMonitor.h\"\n#include \"SetItemOrderDlg.h\"\n#include \"afxdialogex.h\"\n\n\n// CSetItemOrderDlg 对话框\n\nIMPLEMENT_DYNAMIC(CSetItemOrderDlg, CBaseDialog)\n\nCSetItemOrderDlg::CSetItemOrderDlg(CWnd* pParent /*=nullptr*/)\n    : CBaseDialog(IDD_SELECT_ORDER_DIALOG, pParent), m_item_order(true)\n{\n    m_item_order.Init();\n}\n\nCSetItemOrderDlg::~CSetItemOrderDlg()\n{\n}\n\nvoid CSetItemOrderDlg::SetItemOrder(const std::vector<int>& item_order)\n{\n    m_item_order.SetOrder(item_order);\n}\n\nconst std::vector<int>& CSetItemOrderDlg::GetItemOrder() const\n{\n    return m_item_order.GetItemOrderConst();\n}\n\nvoid CSetItemOrderDlg::SetDisplayItem(unsigned int display_item)\n{\n    m_display_item = display_item;\n}\n\nunsigned int CSetItemOrderDlg::GetDisplayItem() const\n{\n    return m_display_item;\n}\n\nvoid CSetItemOrderDlg::SetPluginDisplayItem(const StringSet& plugin_item)\n{\n    m_plugin_item = plugin_item;\n}\n\nconst StringSet& CSetItemOrderDlg::GetPluginDisplayItem() const\n{\n    return m_plugin_item;\n}\n\nvoid CSetItemOrderDlg::DoDataExchange(CDataExchange* pDX)\n{\n    CBaseDialog::DoDataExchange(pDX);\n    DDX_Control(pDX, IDC_LIST1, m_list_ctrl);\n}\n\n\nBEGIN_MESSAGE_MAP(CSetItemOrderDlg, CBaseDialog)\n    ON_BN_CLICKED(IDC_MOVE_UP_BUTTON, &CSetItemOrderDlg::OnBnClickedMoveUpButton)\n    ON_BN_CLICKED(IDC_MOVE_DOWN_BUTTON, &CSetItemOrderDlg::OnBnClickedMoveDownButton)\n    ON_BN_CLICKED(IDC_RESTORE_DEFAULT_BUTTON, &CSetItemOrderDlg::OnBnClickedRestoreDefaultButton)\n    ON_LBN_SELCHANGE(IDC_LIST1, &CSetItemOrderDlg::OnLbnSelchangeList1)\n    ON_CLBN_CHKCHANGE(IDC_LIST1, &CSetItemOrderDlg::OnCheckChanged)\nEND_MESSAGE_MAP()\n\n\n// CSetItemOrderDlg 消息处理程序\n\n\nBOOL CSetItemOrderDlg::OnInitDialog()\n{\n    CBaseDialog::OnInitDialog();\n\n    // TODO:  在此添加额外的初始化\n    SetIcon(theApp.GetMenuIcon(IDI_ITEM), FALSE);\t\t// 设置小图标\n\n    m_list_ctrl.SetItemHeight(0, theApp.DPI(20));\n    EnableCtrl(-1);\n    ShowItem();\n\n    return TRUE;  // return TRUE unless you set the focus to a control\n                  // 异常: OCX 属性页应返回 FALSE\n}\n\nvoid CSetItemOrderDlg::ShowItem()\n{\n    //向列表中添加项目\n    m_list_ctrl.ResetContent();\n    m_all_displayed_item = m_item_order.GetAllDisplayItemsWithOrder();\n    for (const auto& item : m_all_displayed_item)\n    {\n        m_list_ctrl.AddString(CTaskbarItemOrderHelper::GetItemDisplayName(item));\n        if (GetItemChecked(item))\n            m_list_ctrl.SetCheck(m_list_ctrl.GetCount() - 1, TRUE);\n        else\n            m_list_ctrl.SetCheck(m_list_ctrl.GetCount() - 1, FALSE);\n    }\n}\n\nvoid CSetItemOrderDlg::EnableCtrl(int list_selected)\n{\n    EnableDlgCtrl(IDC_MOVE_UP_BUTTON, list_selected > 0 && list_selected < static_cast<int>(m_all_displayed_item.size()));\n    EnableDlgCtrl(IDC_MOVE_DOWN_BUTTON, list_selected >= 0 && list_selected < static_cast<int>(m_all_displayed_item.size()) - 1);\n}\n\nvoid CSetItemOrderDlg::EnableDlgCtrl(UINT id, bool enable)\n{\n    CWnd* pCtrl = GetDlgItem(id);\n    if (pCtrl != nullptr)\n        pCtrl->EnableWindow(enable);\n}\n\nbool CSetItemOrderDlg::GetItemChecked(CommonDisplayItem item)\n{\n    if (item.is_plugin)\n    {\n        if (item.plugin_item != nullptr)\n            return m_plugin_item.Contains(item.plugin_item->GetItemId());\n    }\n    else\n    {\n        return m_display_item & item.item_type;\n    }\n    return false;\n}\n\nvoid CSetItemOrderDlg::SaveItemChecked(CommonDisplayItem item, bool checked)\n{\n    if (item.is_plugin)\n    {\n        if (item.plugin_item != nullptr)\n            m_plugin_item.SetStrContained(item.plugin_item->GetItemId(), checked);\n    }\n    else\n    {\n        if (checked)\n            m_display_item |= item.item_type;\n        else\n            m_display_item &= ~item.item_type;\n    }\n}\n\n\nvoid CSetItemOrderDlg::OnBnClickedMoveUpButton()\n{\n    // TODO: 在此添加控件通知处理程序代码\n    auto& item_list{ m_item_order.GetItemOrder() };\n    int cur_index{ m_list_ctrl.GetCurSel() };\n    int item_count{ static_cast<int>(item_list.size()) };\n    if (cur_index > 0 && cur_index < item_count)\n    {\n        std::swap(item_list[cur_index], item_list[cur_index - 1]);\n        ShowItem();\n        m_list_ctrl.SetCurSel(cur_index - 1);\n        EnableCtrl(cur_index - 1);\n    }\n}\n\n\nvoid CSetItemOrderDlg::OnBnClickedMoveDownButton()\n{\n    // TODO: 在此添加控件通知处理程序代码\n    auto& item_list{ m_item_order.GetItemOrder() };\n    int cur_index{ m_list_ctrl.GetCurSel() };\n    int item_count{ static_cast<int>(item_list.size()) };\n    if (cur_index >= 0 && cur_index < item_count - 1)\n    {\n        std::swap(item_list[cur_index], item_list[cur_index + 1]);\n        ShowItem();\n        m_list_ctrl.SetCurSel(cur_index + 1);\n        EnableCtrl(cur_index + 1);\n    }\n}\n\n\nvoid CSetItemOrderDlg::OnBnClickedRestoreDefaultButton()\n{\n    // TODO: 在此添加控件通知处理程序代码\n    auto item_list{ m_item_order.GetItemOrder() };\n    std::sort(item_list.begin(), item_list.end());\n    m_item_order.SetOrder(item_list);\n    ShowItem();\n}\n\nCString CSetItemOrderDlg::GetDialogName() const\n{\n    return _T(\"SetItemOrderDlg\");\n}\n\n\nvoid CSetItemOrderDlg::OnLbnSelchangeList1()\n{\n    // TODO: 在此添加控件通知处理程序代码\n    int cur_index{ m_list_ctrl.GetCurSel() };\n    EnableCtrl(cur_index);\n}\n\n\nvoid CSetItemOrderDlg::OnCheckChanged()\n{\n    //当用户点击项目前面的复选框时保存该项目的勾选状态\n    int cur_index{ m_list_ctrl.GetCurSel() };\n    if (cur_index >= 0 && cur_index < static_cast<int>(m_all_displayed_item.size()))\n    {\n        bool is_checked = (m_list_ctrl.GetCheck(cur_index) != 0);\n        CommonDisplayItem item = m_all_displayed_item[cur_index];\n        SaveItemChecked(item, is_checked);\n    }\n}\n\nvoid CSetItemOrderDlg::OnOK()\n{\n    // TODO: 在此添加专用代码和/或调用基类\n\n    //保存每个项目的勾选状态\n    //auto item_list = m_item_order.GetAllDisplayItemsWithOrder();\n    //int i = 0;\n    //for (; i < static_cast<int>(item_list.size()); i++)\n    //{\n    //    bool is_checked = (m_list_ctrl.GetCheck(i) != 0);\n    //    CommonDisplayItem item = item_list[i];\n    //    SaveItemChecked(item, is_checked);\n    //}\n\n    if (m_display_item == 0)\n        m_display_item = TDI_UP;\n\n    CBaseDialog::OnOK();\n}\n"
  },
  {
    "path": "TrafficMonitor/SetItemOrderDlg.h",
    "content": "﻿#pragma once\n#include \"BaseDialog.h\"\n\n// CSetItemOrderDlg 对话框\n\nclass CSetItemOrderDlg : public CBaseDialog\n{\n    DECLARE_DYNAMIC(CSetItemOrderDlg)\n\npublic:\n    CSetItemOrderDlg(CWnd* pParent = nullptr);   // 标准构造函数\n    virtual ~CSetItemOrderDlg();\n\n    //设置/获取显示顺序\n    void SetItemOrder(const std::vector<int>& item_order);\n    const std::vector<int>& GetItemOrder() const;\n\n    //设置/获取显示项目，使用unsigned int的每个bit表示对应项目是否显示\n    void SetDisplayItem(unsigned int display_item);\n    unsigned int GetDisplayItem() const;\n\n    void SetPluginDisplayItem(const StringSet& plugin_item);\n    const StringSet& GetPluginDisplayItem() const;\n\n    // 对话框数据\n#ifdef AFX_DESIGN_TIME\n    enum { IDD = IDD_SELECT_ORDER_DIALOG };\n#endif\n\nprivate:\n    CTaskbarItemOrderHelper m_item_order;   //显示项目的顺序\n    CCheckListBox m_list_ctrl;\n    unsigned int m_display_item;        //使用每个bit位表示的要显示的内置项目\n    StringSet m_plugin_item;            //要显示的插件项目\n    std::vector<CommonDisplayItem> m_all_displayed_item;    //在列表中显示的所有项目\n\nprotected:\n    virtual void DoDataExchange(CDataExchange* pDX);    // DDX/DDV 支持\n    void ShowItem();\n    void EnableCtrl(int list_selected);\n    void EnableDlgCtrl(UINT id, bool enable);\n\n    bool GetItemChecked(CommonDisplayItem item);\n    void SaveItemChecked(CommonDisplayItem item, bool checked);\n\n    // 通过 CBaseDialog 继承\n    virtual CString GetDialogName() const override;\n\n    DECLARE_MESSAGE_MAP()\npublic:\n    virtual BOOL OnInitDialog();\n    afx_msg void OnBnClickedMoveUpButton();\n    afx_msg void OnBnClickedMoveDownButton();\n    afx_msg void OnBnClickedRestoreDefaultButton();\n    afx_msg void OnLbnSelchangeList1();\n    afx_msg void OnCheckChanged();\n    virtual void OnOK();\n};\n"
  },
  {
    "path": "TrafficMonitor/SimpleXML.cpp",
    "content": "#include \"stdafx.h\"\n#include \"SimpleXML.h\"\n\n\nCSimpleXML::CSimpleXML(const wstring & xml_path)\n{\n\tifstream file_stream{ xml_path };\n\tif (file_stream.fail())\n\t{\n\t\treturn;\n\t}\n\t//ȡļ\n\tstring xml_str;\n\twhile (!file_stream.eof())\n\t{\n\t\txml_str.push_back(file_stream.get());\n\t}\n\txml_str.pop_back();\n\tif (!xml_str.empty() && xml_str.back() != L'\\n')\t\t//ȷļĩβлس\n\t\txml_str.push_back(L'\\n');\n\t//жļǷutf8\n\tbool is_utf8;\n\tif (xml_str.size() >= 3 && xml_str[0] == -17 && xml_str[1] == -69 && xml_str[2] == -65)\n\t{\n\t\t//UTF8BOMɾBOM\n\t\tis_utf8 = true;\n\t\txml_str = xml_str.substr(3);\n\t}\n\telse\n\t{\n\t\tis_utf8 = false;\n\t}\n\t//תUnicode\n\tm_xml_content = CCommon::StrToUnicode(xml_str.c_str(), is_utf8);\n}\n\nCSimpleXML::CSimpleXML()\n{\n}\n\n\nCSimpleXML::~CSimpleXML()\n{\n}\n\nwstring CSimpleXML::GetNode(const wchar_t * node, const wchar_t * parent) const\n{\n\twstring node_content = _GetNode(parent, m_xml_content);\n\treturn _GetNode(node, node_content);\n}\n\nwstring CSimpleXML::GetNode(const wchar_t * node) const\n{\n\treturn _GetNode(node, m_xml_content);\n}\n\nwstring CSimpleXML::_GetNode(const wchar_t * node, const wstring & content)\n{\n\twstring result;\n\twstring node_start{ L'<' };\n\twstring node_end{ L'<' };\n\tnode_start += node;\n\tnode_start += L'>';\n\tnode_end += L'/';\n\tnode_end += node;\n\tnode_end += L'>';\n\n\tsize_t index_start, index_end;\n\tindex_start = content.find(node_start);\n\tindex_end = content.find(node_end);\n\tif (index_start == wstring::npos || index_end == wstring::npos)\n\t\treturn wstring();\n\n\tresult = content.substr(index_start + node_start.size(), index_end - index_start - node_start.size());\n\treturn result;\n}\n"
  },
  {
    "path": "TrafficMonitor/SimpleXML.h",
    "content": "//һ׵XML\n#pragma once\n#include \"Common.h\"\n\nclass CSimpleXML\n{\npublic:\n\tCSimpleXML(const wstring& xml_path);\n\tCSimpleXML();\n\t~CSimpleXML();\n\n\tvoid LoadXMLContentDirect(const wstring& xml_content) { m_xml_content = xml_content; }\n\n\twstring GetNode(const wchar_t* node, const wchar_t* parent) const;\n\twstring GetNode(const wchar_t* node) const;\n\n\tstatic wstring _GetNode(const wchar_t* node, const wstring& content);\n\nprotected:\n\twstring m_xml_content;\n\n};\n\n"
  },
  {
    "path": "TrafficMonitor/SkinDlg.cpp",
    "content": "﻿// SkinDlg.cpp : 实现文件\n//\n\n#include \"stdafx.h\"\n#include \"TrafficMonitor.h\"\n#include \"SkinDlg.h\"\n#include \"afxdialogex.h\"\n\n\n// CSkinDlg 对话框\n\nIMPLEMENT_DYNAMIC(CSkinDlg, CBaseDialog)\n\nCSkinDlg::CSkinDlg(CWnd* pParent /*=NULL*/)\n    : CBaseDialog(IDD_SKIN_DIALOG, pParent)\n{\n\n}\n\nCSkinDlg::~CSkinDlg()\n{\n}\n\nCString CSkinDlg::GetDialogName() const\n{\n    return _T(\"SkinDlg\");\n}\n\nvoid CSkinDlg::DoDataExchange(CDataExchange* pDX)\n{\n    CBaseDialog::DoDataExchange(pDX);\n    DDX_Control(pDX, IDC_COMBO1, m_select_box);\n    DDX_Control(pDX, IDC_SKIN_COURSE_STATIC, m_skin_course);\n    DDX_Control(pDX, IDC_SKIN_DOWNLOAD_STATIC, m_skin_download);\n    DDX_Control(pDX, IDC_PREVIEW_GROUP_STATIC, m_preview_static);\n    DDX_Control(pDX, IDC_NOTIFY_STATIC, m_notify_static);\n    DDX_Control(pDX, IDC_OPEN_SKIN_DIR_STATIC, m_open_skin_dir_lnk);\n}\n\n\nvoid CSkinDlg::ShowPreview()\n{\n    //载入布局数据\n    wstring cfg_path{ theApp.m_skin_path + m_skins[m_skin_selected] + L\"\\\\skin.xml\" };\n    if (!CCommon::FileExist(cfg_path.c_str()))\n        cfg_path = theApp.m_skin_path + m_skins[m_skin_selected] + L\"\\\\skin.ini\";\n    m_skin_data.Load(cfg_path);\n    //获取预览区大小\n    m_view->SetSize(m_skin_data.GetPreviewInfo().width, m_skin_data.GetPreviewInfo().height);\n    //刷新预览图\n    m_view->Invalidate();\n\n    //显示皮肤作者\n    SetDlgItemText(IDC_SKIN_INFO, CCommon::LoadText(IDS_SKIN_AUTHOUR, m_skin_data.GetSkinInfo().skin_author.c_str()));\n    //设置提示信息\n    bool cover_font_setting{ !m_skin_data.GetSkinInfo().font_info.name.IsEmpty() || (m_skin_data.GetSkinInfo().font_info.size >= MIN_FONT_SIZE && m_skin_data.GetSkinInfo().font_info.size <= MAX_FONT_SIZE) };\n    bool cover_str_setting{ !m_skin_data.GetSkinInfo().display_text.IsInvalid() };\n    cover_font_setting = cover_font_setting && theApp.m_general_data.allow_skin_cover_font;\n    cover_str_setting = cover_str_setting && theApp.m_general_data.allow_skin_cover_text;\n    if (cover_font_setting && cover_str_setting)\n        m_notify_static.SetWindowTextEx(CCommon::LoadText(IDS_OVERWRITE_FONT_TEXT_WARNING));\n    else if (cover_font_setting)\n        m_notify_static.SetWindowTextEx(CCommon::LoadText(IDS_OVERWRITE_FONT_WARNING));\n    else if (cover_str_setting)\n        m_notify_static.SetWindowTextEx(CCommon::LoadText(IDS_OVERWRITE_TEXT_WARNING));\n    else\n        m_notify_static.SetWindowTextEx(_T(\"\"));\n}\n\n\nCRect CSkinDlg::CalculateViewRect()\n{\n    CRect rect;\n    m_preview_static.GetWindowRect(rect);\t\t//获取“预览” group box 的位置\n    ScreenToClient(&rect);\n    CRect scroll_view_rect{ rect };\n    scroll_view_rect.DeflateRect(theApp.DPI(12), theApp.DPI(40));\n    scroll_view_rect.top = rect.top + theApp.DPI(28);\n    return scroll_view_rect;\n}\n\n\nBEGIN_MESSAGE_MAP(CSkinDlg, CBaseDialog)\n    ON_CBN_SELCHANGE(IDC_COMBO1, &CSkinDlg::OnCbnSelchangeCombo1)\n    ON_WM_SIZE()\n    ON_MESSAGE(WM_LINK_CLICKED, &CSkinDlg::OnLinkClicked)\nEND_MESSAGE_MAP()\n\n\n// CSkinDlg 消息处理程序\n\nBOOL CSkinDlg::OnInitDialog()\n{\n    CBaseDialog::OnInitDialog();\n\n    // TODO:  在此添加额外的初始化\n    SetWindowText(CCommon::LoadText(IDS_TITLE_CHANGE_SKIN));\n    SetIcon(theApp.GetMenuIcon(IDI_SKIN), FALSE);\t\t// 设置小图标\n    //初始化选择框\n    for (const auto& skin_path : m_skins)\n    {\n        wstring skin_name;\n        size_t index = skin_path.find_last_of(L'\\\\');\n        skin_name = skin_path.substr(index + 1);\n        m_select_box.AddString(skin_name.c_str());\n    }\n    m_select_box.SetCurSel(m_skin_selected);\n    m_select_box.SetMinVisibleItems(9);\n    //初始化预览视图\n    m_view = (CSkinPreviewView*)RUNTIME_CLASS(CSkinPreviewView)->CreateObject();\n    m_view->Create(NULL, NULL, WS_CHILD | WS_VISIBLE | WS_VSCROLL | WS_HSCROLL, CalculateViewRect(), this, 3000);\n    m_view->InitialUpdate();\n    m_view->SetSkinData(&m_skin_data);\n    m_view->SetFont(m_pFont);\n    m_view->ShowWindow(SW_SHOW);\n\n    //设置提示信息\n    m_notify_static.SetTextColor(RGB(252, 128, 45));\n    m_notify_static.SetBackColor(GetSysColor(COLOR_BTNFACE));\n    m_notify_static.SetWindowTextEx(_T(\"\"));\n\n    //显示预览图片\n    ShowPreview();\n\n    //设置超链接\n    m_skin_course.SetURL(_T(\"https://github.com/zhongyang219/TrafficMonitor/wiki/%E7%9A%AE%E8%82%A4%E5%88%B6%E4%BD%9C%E6%95%99%E7%A8%8B\"));\n    m_skin_download.SetURL(_T(\"https://github.com/zhongyang219/TrafficMonitorSkin/blob/master/皮肤下载.md\"));\n    m_open_skin_dir_lnk.SetLinkIsURL(false);\n\n    return TRUE;  // return TRUE unless you set the focus to a control\n                  // 异常: OCX 属性页应返回 FALSE\n}\n\n\nvoid CSkinDlg::OnCbnSelchangeCombo1()\n{\n    // TODO: 在此添加控件通知处理程序代码\n    m_skin_selected = m_select_box.GetCurSel();\n    ShowPreview();\n}\n\n\nvoid CSkinDlg::OnSize(UINT nType, int cx, int cy)\n{\n    CBaseDialog::OnSize(nType, cx, cy);\n\n    // TODO: 在此处添加消息处理程序代码\n    if (m_preview_static.m_hWnd != NULL && nType != SIZE_MINIMIZED && m_view != nullptr)\n        m_view->MoveWindow(CalculateViewRect());\n}\n\n\nafx_msg LRESULT CSkinDlg::OnLinkClicked(WPARAM wParam, LPARAM lParam)\n{\n    CWnd* pCtrl = (CWnd*)wParam;\n    //点击了“打开皮肤目录”\n    if (pCtrl == &m_open_skin_dir_lnk)\n    {\n        CreateDirectory(theApp.m_skin_path.c_str(), NULL);       //如果皮肤目录不存在，则创建它\n        ShellExecute(NULL, _T(\"open\"), _T(\"explorer\"), theApp.m_skin_path.c_str(), NULL, SW_SHOWNORMAL);\n    }\n    return 0;\n}\n"
  },
  {
    "path": "TrafficMonitor/SkinDlg.h",
    "content": "﻿#pragma once\n#include \"afxwin.h\"\n#include\"StaticEx.h\"\n#include \"PictureStatic.h\"\n#include \"CSkinPreviewView.h\"\n#include \"LinkStatic.h\"\n#include \"BaseDialog.h\"\n\n// CSkinDlg 对话框\n\nclass CSkinDlg : public CBaseDialog\n{\n    DECLARE_DYNAMIC(CSkinDlg)\n\npublic:\n    CSkinDlg(CWnd* pParent = NULL);   // 标准构造函数\n    virtual ~CSkinDlg();\n\n    vector<wstring> m_skins;\t\t//皮肤文件的路径\n    int m_skin_selected;\t\t\t//选择的皮肤\n    CFont* m_pFont;\t\t//预览图的字体\n\n// 对话框数据\n#ifdef AFX_DESIGN_TIME\n    enum { IDD = IDD_SKIN_DIALOG };\n#endif\n\n    const CSkinFile& GetSkinData() const { return m_skin_data; }\n\nprotected:\n    virtual CString GetDialogName() const override;\n    virtual void DoDataExchange(CDataExchange* pDX);    // DDX/DDV 支持\n\nprotected:\n    //控件变量\n    CComboBox m_select_box;\t\t\t//选择框\n    CLinkStatic m_skin_course;\t//“皮肤制作教程”超链接\n    CLinkStatic m_skin_download;\t//“更多皮肤下载”超链接\n    CLinkStatic m_open_skin_dir_lnk;\n    CSkinPreviewView* m_view;\t//预览区视图类\n    CStatic m_preview_static;\n    CStaticEx m_notify_static;\t//显示提示信息的static控件\n\n    CSkinFile m_skin_data;\t\t//皮肤数据\n\n    void ShowPreview();\t\t//显示皮肤预览\n    CRect CalculateViewRect();\t\t//根据窗口大小计算预览视图的大小\n\n    DECLARE_MESSAGE_MAP()\npublic:\n    virtual BOOL OnInitDialog();\n    afx_msg void OnCbnSelchangeCombo1();\n    afx_msg void OnSize(UINT nType, int cx, int cy);\nprotected:\n    afx_msg LRESULT OnLinkClicked(WPARAM wParam, LPARAM lParam);\n};\n"
  },
  {
    "path": "TrafficMonitor/SkinFile.cpp",
    "content": "﻿#include \"stdafx.h\"\n#include \"SkinFile.h\"\n#include \"Common.h\"\n#include \"FilePathHelper.h\"\n#include \"TrafficMonitor.h\"\n#include \"IniHelper.h\"\n#include \"DrawCommon.h\"\n\n\nCSkinFile::CSkinFile()\n{\n}\n\n\nCSkinFile::~CSkinFile()\n{\n}\n\nstatic CSkinFile::LayoutItem LayoutItemFromXmlNode(tinyxml2::XMLElement* ele)\n{\n    CSkinFile::LayoutItem layout_item;\n    layout_item.x = theApp.DPI(atoi(CTinyXml2Helper::ElementAttribute(ele, \"x\")));\n    layout_item.y = theApp.DPI(atoi(CTinyXml2Helper::ElementAttribute(ele, \"y\")));\n    layout_item.width = theApp.DPI(atoi(CTinyXml2Helper::ElementAttribute(ele, \"width\")));\n    layout_item.align = static_cast<Alignment>(atoi(CTinyXml2Helper::ElementAttribute(ele, \"align\")));\n    layout_item.show = CTinyXml2Helper::StringToBool(CTinyXml2Helper::ElementAttribute(ele, \"show\"));\n    return layout_item;\n}\n\nCSkinFile::Layout CSkinFile::LayoutFromXmlNode(tinyxml2::XMLElement* ele)\n{\n    CSkinFile::Layout layout;\n    layout.width = theApp.DPI(atoi(CTinyXml2Helper::ElementAttribute(ele, \"width\")));\n    layout.height = theApp.DPI(atoi(CTinyXml2Helper::ElementAttribute(ele, \"height\")));\n    CTinyXml2Helper::IterateChildNode(ele, [&](tinyxml2::XMLElement* ele_layout_item)\n        {\n            string str_layout_item = CTinyXml2Helper::ElementName(ele_layout_item);\n            for (auto display_item : AllDisplayItems)\n            {\n                if (str_layout_item == CSkinFile::GetDisplayItemXmlNodeName(display_item))\n                {\n                    layout.layout_items[display_item] = LayoutItemFromXmlNode(ele_layout_item);\n                    break;\n                }\n            }\n            wstring plugin_id = CCommon::StrToUnicode(m_plugin_map[str_layout_item].c_str(), true);\n            if (!plugin_id.empty())\n            {\n                for (const auto& plugin_item : theApp.m_plugins.GetPluginItems())\n                {\n                    if (plugin_id == plugin_item->GetItemId())\n                    {\n                        layout.layout_items[plugin_item] = LayoutItemFromXmlNode(ele_layout_item);\n                        break;\n                    }\n                }\n            }\n        });\n    return layout;\n}\n\nvoid CSkinFile::DrawSkinText(CDrawCommon drawer, DrawStr draw_str, CRect rect, COLORREF color, Alignment align)\n{\n    int text_width = drawer.GetDC()->GetTextExtent(draw_str.GetStr()).cx;\n    //标签和数值两端对齐\n    if (align == Alignment::SIDE && text_width < rect.Width())      //只有文本宽度小于矩形的宽度时才使用两端对齐\n    {\n        //绘制标签\n        drawer.DrawWindowText(rect, draw_str.label, color, Alignment::LEFT);\n        //绘制数值\n        drawer.DrawWindowText(rect, draw_str.value, color, Alignment::RIGHT);\n    }\n    else\n    {\n        drawer.DrawWindowText(rect, draw_str.GetStr(), color, align);\n    }\n}\n\nvoid CSkinFile::Load(const wstring& file_path)\n{\n    CFilePathHelper file_path_helper{ file_path };\n    wstring ext = file_path_helper.GetFileExtension();\n    if (ext == L\"ini\")\n        LoadFromIni(file_path);\n    else\n        LoadFromXml(file_path);\n\n    if (m_font.m_hObject)\n        m_font.DeleteObject();\n\n    //创建字体对象\n    m_skin_info.font_info.Create(m_font);\n\n    //载入背景图片\n    wstring path_dir = file_path_helper.GetDir();\n    m_background_s.Destroy();\n    m_background_s.Load((path_dir + BACKGROUND_IMAGE_S).c_str());\n    m_background_l.Destroy();\n    m_background_l.Load((path_dir + BACKGROUND_IMAGE_L).c_str());\n}\n\nvoid CSkinFile::LoadFromXml(const wstring& file_path)\n{\n    m_skin_info = SkinInfo();\n    m_layout_info = LayoutInfo();\n    m_preview_info = PreviewInfo();\n\n    tinyxml2::XMLDocument doc;\n    if (CTinyXml2Helper::LoadXmlFile(doc, file_path.c_str()))\n    {\n        CTinyXml2Helper::IterateChildNode(doc.FirstChildElement(), [this](tinyxml2::XMLElement* child)\n            {\n                string ele_name = CTinyXml2Helper::ElementName(child);\n                //读取皮肤信息\n                if (ele_name == \"skin\")\n                {\n                    CTinyXml2Helper::IterateChildNode(child, [this](tinyxml2::XMLElement* skin_item)\n                        {\n                            string skin_item_name = CTinyXml2Helper::ElementName(skin_item);\n                            //文本颜色\n                            if (skin_item_name == \"text_color\")\n                            {\n                                string str_text_color = CTinyXml2Helper::ElementText(skin_item);\n                                std::vector<string> split_result;\n                                CCommon::StringSplit(str_text_color, L',', split_result);\n                                for (const auto& str : split_result)\n                                {\n                                    m_skin_info.text_color.push_back(atoi(str.c_str()));\n                                }\n                            }\n\n                            if (m_skin_info.text_color.size() < theApp.m_plugins.AllDisplayItemsWithPlugins().size())\n                            {\n                                COLORREF default_color{};\n                                if (!m_skin_info.text_color.empty())\n                                    default_color = m_skin_info.text_color.front();\n                                m_skin_info.text_color.resize(theApp.m_plugins.AllDisplayItemsWithPlugins().size(), default_color);\n                            }\n                            //指定每个项目的颜色\n                            else if (skin_item_name == \"specify_each_item_color\")\n                            {\n                                m_skin_info.specify_each_item_color = CTinyXml2Helper::StringToBool(CTinyXml2Helper::ElementText(skin_item));\n                            }\n                            //皮肤作者\n                            else if (skin_item_name == \"skin_author\")\n                            {\n                                m_skin_info.skin_author = CCommon::StrToUnicode(CTinyXml2Helper::ElementText(skin_item), true);\n                            }\n                            //字体\n                            else if (skin_item_name == \"font\")\n                            {\n                                m_skin_info.font_info.name = CTinyXml2Helper::ElementAttribute(skin_item, \"name\");\n                                m_skin_info.font_info.size = atoi(CTinyXml2Helper::ElementAttribute(skin_item, \"size\"));\n                                int font_style = atoi(CTinyXml2Helper::ElementAttribute(skin_item, \"style\"));\n                                m_skin_info.font_info.bold = CCommon::GetNumberBit(font_style, 0);\n                                m_skin_info.font_info.italic = CCommon::GetNumberBit(font_style, 1);\n                                m_skin_info.font_info.underline = CCommon::GetNumberBit(font_style, 2);\n                                m_skin_info.font_info.strike_out = CCommon::GetNumberBit(font_style, 3);\n                            }\n                            else if (skin_item_name == \"display_text\")\n                            {\n                                CTinyXml2Helper::IterateChildNode(skin_item, [this](tinyxml2::XMLElement* display_text_item)\n                                    {\n                                        string item_name = CTinyXml2Helper::ElementName(display_text_item);\n                                        wstring item_text = CCommon::StrToUnicode(CTinyXml2Helper::ElementText(display_text_item), true);\n                                        for (auto display_item : AllDisplayItems)\n                                        {\n                                            if (item_name == CSkinFile::GetDisplayItemXmlNodeName(display_item))\n                                            {\n                                                m_skin_info.display_text.Get(display_item) = item_text;\n                                                break;\n                                            }\n                                        }\n                                    });\n                            }\n                        });\n                }\n                //布局信息\n                else if (ele_name == \"layout\")\n                {\n                    m_layout_info.text_height = theApp.DPI(atoi(CTinyXml2Helper::ElementAttribute(child, \"text_height\")));\n                    m_layout_info.no_label = CTinyXml2Helper::StringToBool(CTinyXml2Helper::ElementAttribute(child, \"no_label\"));\n                    CTinyXml2Helper::IterateChildNode(child, [this](tinyxml2::XMLElement* ele_layout)\n                        {\n                            string str_layout = CTinyXml2Helper::ElementName(ele_layout);\n                            if (str_layout == \"layout_l\")\n                                m_layout_info.layout_l = LayoutFromXmlNode(ele_layout);\n                            else if (str_layout == \"layout_s\")\n                                m_layout_info.layout_s = LayoutFromXmlNode(ele_layout);\n                        });\n                }\n                //预览图\n                else if (ele_name == \"preview\")\n                {\n                    m_preview_info.width = theApp.DPI(atoi(CTinyXml2Helper::ElementAttribute(child, \"width\")));\n                    m_preview_info.height = theApp.DPI(atoi(CTinyXml2Helper::ElementAttribute(child, \"height\")));\n                    CTinyXml2Helper::IterateChildNode(child, [this](tinyxml2::XMLElement* ele_priview_item)\n                        {\n                            string str_item_name = CTinyXml2Helper::ElementName(ele_priview_item);\n                            if (str_item_name == \"l\")\n                            {\n                                m_preview_info.l_pos.x = theApp.DPI(atoi(CTinyXml2Helper::ElementAttribute(ele_priview_item, \"x\")));\n                                m_preview_info.l_pos.y = theApp.DPI(atoi(CTinyXml2Helper::ElementAttribute(ele_priview_item, \"y\")));\n                            }\n                            else if (str_item_name == \"s\")\n                            {\n                                m_preview_info.s_pos.x = theApp.DPI(atoi(CTinyXml2Helper::ElementAttribute(ele_priview_item, \"x\")));\n                                m_preview_info.s_pos.y = theApp.DPI(atoi(CTinyXml2Helper::ElementAttribute(ele_priview_item, \"y\")));\n                            }\n                        });\n                }\n                //插件名称映射\n                else if (ele_name == \"plugin_map\")\n                {\n                    CTinyXml2Helper::IterateChildNode(child, [this](tinyxml2::XMLElement* plugin_item)\n                        {\n                            string ele_name = CTinyXml2Helper::ElementName(plugin_item);\n                            string ele_text = CTinyXml2Helper::ElementText(plugin_item);\n                            m_plugin_map[ele_name] = ele_text;\n                        });\n                }\n            });\n    }\n\n}\n\nvoid CSkinFile::LoadFromIni(const wstring& file_path)\n{\n    m_skin_info = SkinInfo();\n    m_layout_info = LayoutInfo();\n    m_preview_info = PreviewInfo();\n\n    //获取皮肤信息\n    CIniHelper ini(file_path);\n    //获取当前皮肤的文字颜色\n    std::map<CommonDisplayItem, COLORREF> text_colors{};\n    ini.LoadMainWndColors(_T(\"skin\"), _T(\"text_color\"), text_colors, 0);\n    for (const auto& item : text_colors)\n    {\n        m_skin_info.text_color.push_back(item.second);\n    }\n\n    m_skin_info.specify_each_item_color = ini.GetBool(_T(\"skin\"), _T(\"specify_each_item_color\"), false);\n    //获取当前皮肤的字体\n    FontInfo default_font{};\n    ini.LoadFontData(L\"skin\", m_skin_info.font_info, default_font);\n    //获取皮肤作者\n    m_skin_info.skin_author = ini.GetString(_T(\"skin\"), _T(\"skin_author\"), _T(\"unknow\"));\n    //获取显示文本\n    m_skin_info.display_text.Get(TDI_UP) = ini.GetString(_T(\"skin\"), _T(\"up_string\"), NONE_STR);\n    m_skin_info.display_text.Get(TDI_DOWN) = ini.GetString(_T(\"skin\"), _T(\"down_string\"), NONE_STR);\n    m_skin_info.display_text.Get(TDI_CPU) = ini.GetString(_T(\"skin\"), _T(\"cpu_string\"), NONE_STR);\n    m_skin_info.display_text.Get(TDI_MEMORY) = ini.GetString(_T(\"skin\"), _T(\"memory_string\"), NONE_STR);\n    //获取预览区大小\n    m_preview_info.width = theApp.DPI(ini.GetInt(_T(\"layout\"), _T(\"preview_width\"), 238));\n    m_preview_info.height = theApp.DPI(ini.GetInt(_T(\"layout\"), _T(\"preview_height\"), 105));\n\n    //从ini文件读取皮肤布局，并根据DPI进行缩放\n    m_layout_info.text_height = theApp.DPI(ini.GetInt(_T(\"layout\"), _T(\"text_height\"), 20));\n    m_layout_info.no_label = ini.GetBool(_T(\"layout\"), _T(\"no_text\"), false);\n\n    m_layout_info.layout_l.width = theApp.DPI(ini.GetInt(_T(\"layout\"), _T(\"width_l\"), 220));\n    m_layout_info.layout_l.height = theApp.DPI(ini.GetInt(_T(\"layout\"), _T(\"height_l\"), 43));\n    m_layout_info.layout_l.layout_items[TDI_UP].x = theApp.DPI(ini.GetInt(_T(\"layout\"), _T(\"up_x_l\"), 6));\n    m_layout_info.layout_l.layout_items[TDI_UP].y = theApp.DPI(ini.GetInt(_T(\"layout\"), _T(\"up_y_l\"), 2));\n    m_layout_info.layout_l.layout_items[TDI_UP].width = theApp.DPI(ini.GetInt(_T(\"layout\"), _T(\"up_width_l\"), 108));\n    m_layout_info.layout_l.layout_items[TDI_UP].align = static_cast<Alignment>(ini.GetInt(_T(\"layout\"), _T(\"up_align_l\"), 0));\n    m_layout_info.layout_l.layout_items[TDI_DOWN].x = theApp.DPI(ini.GetInt(_T(\"layout\"), _T(\"down_x_l\"), 114));\n    m_layout_info.layout_l.layout_items[TDI_DOWN].y = theApp.DPI(ini.GetInt(_T(\"layout\"), _T(\"down_y_l\"), 2));\n    m_layout_info.layout_l.layout_items[TDI_DOWN].width = theApp.DPI(ini.GetInt(_T(\"layout\"), _T(\"down_width_l\"), 110));\n    m_layout_info.layout_l.layout_items[TDI_DOWN].align = static_cast<Alignment>(ini.GetInt(_T(\"layout\"), _T(\"down_align_l\"), 0));\n    m_layout_info.layout_l.layout_items[TDI_CPU].x = theApp.DPI(ini.GetInt(_T(\"layout\"), _T(\"cpu_x_l\"), 6));\n    m_layout_info.layout_l.layout_items[TDI_CPU].y = theApp.DPI(ini.GetInt(_T(\"layout\"), _T(\"cpu_y_l\"), 21));\n    m_layout_info.layout_l.layout_items[TDI_CPU].width = theApp.DPI(ini.GetInt(_T(\"layout\"), _T(\"cpu_width_l\"), 108));\n    m_layout_info.layout_l.layout_items[TDI_CPU].align = static_cast<Alignment>(ini.GetInt(_T(\"layout\"), _T(\"cpu_align_l\"), 0));\n    m_layout_info.layout_l.layout_items[TDI_MEMORY].x = theApp.DPI(ini.GetInt(_T(\"layout\"), _T(\"memory_x_l\"), 114));\n    m_layout_info.layout_l.layout_items[TDI_MEMORY].y = theApp.DPI(ini.GetInt(_T(\"layout\"), _T(\"memory_y_l\"), 21));\n    m_layout_info.layout_l.layout_items[TDI_MEMORY].width = theApp.DPI(ini.GetInt(_T(\"layout\"), _T(\"memory_width_l\"), 110));\n    m_layout_info.layout_l.layout_items[TDI_MEMORY].align = static_cast<Alignment>(ini.GetInt(_T(\"layout\"), _T(\"memory_align_l\"), 0));\n    m_layout_info.layout_l.layout_items[TDI_UP].show = ini.GetBool(_T(\"layout\"), _T(\"show_up_l\"), true);\n    m_layout_info.layout_l.layout_items[TDI_DOWN].show = ini.GetBool(_T(\"layout\"), _T(\"show_down_l\"), true);\n    m_layout_info.layout_l.layout_items[TDI_CPU].show = ini.GetBool(_T(\"layout\"), _T(\"show_cpu_l\"), true);\n    m_layout_info.layout_l.layout_items[TDI_MEMORY].show = ini.GetBool(_T(\"layout\"), _T(\"show_memory_l\"), true);\n    m_preview_info.l_pos.x = theApp.DPI(ini.GetInt(_T(\"layout\"), _T(\"preview_x_l\"), 0));\n    m_preview_info.l_pos.y = theApp.DPI(ini.GetInt(_T(\"layout\"), _T(\"preview_y_l\"), 47));\n\n    m_layout_info.layout_s.width = theApp.DPI(ini.GetInt(_T(\"layout\"), _T(\"width_s\"), 220));\n    m_layout_info.layout_s.height = theApp.DPI(ini.GetInt(_T(\"layout\"), _T(\"height_s\"), 28));\n    m_layout_info.layout_s.layout_items[TDI_UP].x = theApp.DPI(ini.GetInt(_T(\"layout\"), _T(\"up_x_s\"), 6));\n    m_layout_info.layout_s.layout_items[TDI_UP].y = theApp.DPI(ini.GetInt(_T(\"layout\"), _T(\"up_y_s\"), 4));\n    m_layout_info.layout_s.layout_items[TDI_UP].width = theApp.DPI(ini.GetInt(_T(\"layout\"), _T(\"up_width_s\"), 108));\n    m_layout_info.layout_s.layout_items[TDI_UP].align = static_cast<Alignment>(ini.GetInt(_T(\"layout\"), _T(\"up_align_s\"), 0));\n    m_layout_info.layout_s.layout_items[TDI_DOWN].x = theApp.DPI(ini.GetInt(_T(\"layout\"), _T(\"down_x_s\"), 114));\n    m_layout_info.layout_s.layout_items[TDI_DOWN].y = theApp.DPI(ini.GetInt(_T(\"layout\"), _T(\"down_y_s\"), 4));\n    m_layout_info.layout_s.layout_items[TDI_DOWN].width = theApp.DPI(ini.GetInt(_T(\"layout\"), _T(\"down_width_s\"), 110));\n    m_layout_info.layout_s.layout_items[TDI_DOWN].align = static_cast<Alignment>(ini.GetInt(_T(\"layout\"), _T(\"down_align_s\"), 0));\n    m_layout_info.layout_s.layout_items[TDI_CPU].x = theApp.DPI(ini.GetInt(_T(\"layout\"), _T(\"cpu_x_s\"), 0));\n    m_layout_info.layout_s.layout_items[TDI_CPU].y = theApp.DPI(ini.GetInt(_T(\"layout\"), _T(\"cpu_y_s\"), 0));\n    m_layout_info.layout_s.layout_items[TDI_CPU].width = theApp.DPI(ini.GetInt(_T(\"layout\"), _T(\"cpu_width_s\"), 0));\n    m_layout_info.layout_s.layout_items[TDI_CPU].align = static_cast<Alignment>(ini.GetInt(_T(\"layout\"), _T(\"cpu_align_s\"), 0));\n    m_layout_info.layout_s.layout_items[TDI_MEMORY].x = theApp.DPI(ini.GetInt(_T(\"layout\"), _T(\"memory_x_s\"), 0));\n    m_layout_info.layout_s.layout_items[TDI_MEMORY].y = theApp.DPI(ini.GetInt(_T(\"layout\"), _T(\"memory_y_s\"), 0));\n    m_layout_info.layout_s.layout_items[TDI_MEMORY].width = theApp.DPI(ini.GetInt(_T(\"layout\"), _T(\"memory_width_s\"), 0));\n    m_layout_info.layout_s.layout_items[TDI_MEMORY].align = static_cast<Alignment>(ini.GetInt(_T(\"layout\"), _T(\"memory_align_s\"), 0));\n    m_layout_info.layout_s.layout_items[TDI_UP].show = ini.GetBool(_T(\"layout\"), _T(\"show_up_s\"), true);\n    m_layout_info.layout_s.layout_items[TDI_DOWN].show = ini.GetBool(_T(\"layout\"), _T(\"show_down_s\"), true);\n    m_layout_info.layout_s.layout_items[TDI_CPU].show = ini.GetBool(_T(\"layout\"), _T(\"show_cpu_s\"), false);\n    m_layout_info.layout_s.layout_items[TDI_MEMORY].show = ini.GetBool(_T(\"layout\"), _T(\"show_memory_s\"), false);\n    m_preview_info.s_pos.x = theApp.DPI(ini.GetInt(_T(\"layout\"), _T(\"preview_x_s\"), 0));\n    m_preview_info.s_pos.y = theApp.DPI(ini.GetInt(_T(\"layout\"), _T(\"preview_y_s\"), 0));\n}\n\n\nvoid CSkinFile::DrawPreview(CDC* pDC, CRect rect)\n{\n    CDrawCommon draw;\n    draw.Create(pDC, nullptr);\n    if (!m_skin_info.font_info.name.IsEmpty() && m_skin_info.font_info.size > 0)\n        draw.SetFont(&m_font);\n    else\n        draw.SetFont(theApp.m_pMainWnd->GetFont());\n    draw.SetDrawRect(rect);\n    draw.FillRect(rect, RGB(255, 255, 255));\n    //绘制背景\n    CRect rect_s(CPoint(m_preview_info.s_pos.x, m_preview_info.s_pos.y), CSize(m_layout_info.layout_s.width, m_layout_info.layout_s.height));\n    CRect rect_l(CPoint(m_preview_info.l_pos.x, m_preview_info.l_pos.y), CSize(m_layout_info.layout_l.width, m_layout_info.layout_l.height));\n    if (m_background_s.IsNull())\n        draw.FillRect(rect_s, RGB(230, 230, 230));\n    else\n        draw.DrawBitmap(m_background_s, rect_s.TopLeft(), rect_s.Size());\n    if (m_background_l.IsNull())\n        draw.FillRect(rect_l, RGB(230, 230, 230));\n    draw.DrawBitmap(m_background_l, rect_l.TopLeft(), rect_l.Size());\n\n    //获取每个项目显示的文本\n    std::map<DisplayItem, DrawStr> map_str;\n    for (auto iter = AllDisplayItems.begin(); iter != AllDisplayItems.end(); ++iter)\n    {\n        //wstring disp_text = m_skin_info.display_text.Get(*iter);\n        //if (disp_text == NONE_STR)\n        //    disp_text = theApp.m_main_wnd_data.disp_str.Get(*iter);\n        DrawStr draw_str;\n        switch (*iter)\n        {\n        case TDI_UP:\n            draw_str.value = _T(\"88.8 KB/s\");\n            break;\n        case TDI_DOWN:\n            draw_str.value = _T(\"88.9 KB/s\");\n            break;\n        case TDI_TOTAL_SPEED:\n            draw_str.value = _T(\"90 KB/s\");\n            break;\n        case TDI_CPU:\n            draw_str.value = _T(\"50 %\");\n            break;\n        case TDI_MEMORY:\n            draw_str.value = _T(\"51 %\");\n            break;\n        case TDI_CPU_TEMP: case TDI_GPU_TEMP: case TDI_HDD_TEMP: case TDI_MAIN_BOARD_TEMP:\n            draw_str.value = _T(\"40 °C\");\n            break;\n        case TDI_CPU_FREQ:\n            draw_str.value = _T(\"1.0 GHz\");\n            break;\n        default:\n            draw_str.value = _T(\"99\");\n            break;\n        }\n        if (m_skin_info.display_text.Get(*iter) == NONE_STR)\n            m_skin_info.display_text.Get(*iter) = theApp.m_main_wnd_data.disp_str.Get(*iter);\n        if (!m_layout_info.no_label)\n            draw_str.label = m_skin_info.display_text.Get(*iter).c_str();\n        map_str[*iter] = draw_str;\n    }\n\n    //获取文本颜色\n    std::map<CommonDisplayItem, COLORREF> text_colors{};\n    if (m_skin_info.specify_each_item_color)\n    {\n        int i{};\n        for (const auto& item : theApp.m_plugins.AllDisplayItemsWithPlugins())\n        {\n            if (i < static_cast<int>(m_skin_info.text_color.size()))\n                text_colors[item] = m_skin_info.text_color[i];\n            i++;\n        }\n    }\n    else if (!m_skin_info.text_color.empty())\n    {\n        for (const auto& item : theApp.m_plugins.AllDisplayItemsWithPlugins())\n        {\n            if (!m_skin_info.text_color.empty())\n                text_colors[item] = m_skin_info.text_color[0];\n        }\n    }\n\n    //绘制预览图文本\n    auto drawPreviewText = [&](Layout& layout, const PreviewInfo::Pos& pos)\n    {\n        for (auto iter = map_str.begin(); iter != map_str.end(); ++iter)\n        {\n            if (layout.layout_items[iter->first].show)\n            {\n                CPoint point;\n                point.SetPoint(layout.layout_items[iter->first].x, layout.layout_items[iter->first].y);\n                point.Offset(pos.x, pos.y);\n                CRect rect(point, CSize(layout.layout_items[iter->first].width, m_layout_info.text_height));\n                COLORREF text_color{};\n                text_color = text_colors[iter->first];\n                DrawSkinText(draw, iter->second, rect, text_color, layout.layout_items[iter->first].align);\n            }\n        }\n\n        //绘制插件项目\n        for (const auto& plugin_item : theApp.m_plugins.GetPluginItems())\n        {\n            LayoutItem layout_item = layout.GetItem(plugin_item);\n            if (layout_item.show)\n            {\n                COLORREF cl{};\n                auto iter = text_colors.find(plugin_item);\n                if (iter != text_colors.end())\n                    cl = iter->second;\n                else if (!text_colors.empty())\n                    cl = text_colors.begin()->second;\n                //矩形区域\n                CPoint point;\n                point.SetPoint(layout_item.x, layout_item.y);\n                point.Offset(pos.x, pos.y);\n                CRect rect(point, CSize(layout_item.width, m_layout_info.text_height));\n                if (plugin_item->IsCustomDraw())\n                {\n                    int brightness{ (GetRValue(cl) + GetGValue(cl) + GetBValue(cl)) / 2 };\n                    ITMPlugin* plugin = theApp.m_plugins.GetPluginByItem(plugin_item);\n                    if (plugin != nullptr && plugin->GetAPIVersion() >= 2)\n                    {\n                        plugin->OnExtenedInfo(ITMPlugin::EI_VALUE_TEXT_COLOR, std::to_wstring(cl).c_str());\n                        plugin->OnExtenedInfo(ITMPlugin::EI_DRAW_TASKBAR_WND, L\"0\");\n                    }\n                    draw.GetDC()->SetTextColor(cl);\n                    plugin_item->DrawItem(draw.GetDC()->GetSafeHdc(), point.x, point.y, layout_item.width, m_layout_info.text_height, brightness >= 128);\n                }\n                else\n                {\n                    //绘制文本\n                    DrawStr draw_str;\n                    draw_str.label = plugin_item->GetItemLableText();\n                    draw_str.value = plugin_item->GetItemValueSampleText();\n                    DrawSkinText(draw, draw_str, rect, cl, layout_item.align);\n                }\n            }\n        }\n\n    };\n\n    //绘制小预览图文本\n    drawPreviewText(m_layout_info.layout_s, m_preview_info.s_pos);\n    //绘制大预览图文本\n    drawPreviewText(m_layout_info.layout_l, m_preview_info.l_pos);\n}\n\nvoid CSkinFile::DrawInfo(CDC* pDC, bool show_more_info, CFont& font)\n{\n    //绘制背景图\n    CImage& background_image{ show_more_info ? m_background_l : m_background_s };\n    Layout& layout{ show_more_info ? m_layout_info.layout_l : m_layout_info.layout_s };\n\n    CRect rect(CPoint(0, 0), CSize(layout.width, layout.height));\n    CDrawDoubleBuffer draw_double_buffer(pDC, rect);\n    CDrawCommon draw;\n    draw.Create(draw_double_buffer.GetMemDC(), nullptr);\n\n    draw.DrawBitmap(background_image, CPoint(0, 0), CSize(layout.width, layout.height));\n\n    //获取每个项目显示的文本\n    std::map<DisplayItem, DrawStr> map_str;\n    if (!m_layout_info.no_label)\n    {\n        map_str[TDI_UP].label = theApp.m_main_wnd_data.disp_str.Get(TDI_UP).c_str();\n        map_str[TDI_DOWN].label = theApp.m_main_wnd_data.disp_str.Get(TDI_DOWN).c_str();\n        map_str[TDI_TOTAL_SPEED].label = theApp.m_main_wnd_data.disp_str.Get(TDI_TOTAL_SPEED).c_str();\n        map_str[TDI_CPU].label = theApp.m_main_wnd_data.disp_str.Get(TDI_CPU).c_str();\n        map_str[TDI_MEMORY].label = theApp.m_main_wnd_data.disp_str.Get(TDI_MEMORY).c_str();\n        map_str[TDI_GPU_USAGE].label = theApp.m_main_wnd_data.disp_str.Get(TDI_GPU_USAGE).c_str();\n        map_str[TDI_HDD_USAGE].label = theApp.m_main_wnd_data.disp_str.Get(TDI_HDD_USAGE).c_str();\n        map_str[TDI_CPU_TEMP].label = theApp.m_main_wnd_data.disp_str.Get(TDI_CPU_TEMP).c_str();\n        map_str[TDI_CPU_FREQ].label = theApp.m_main_wnd_data.disp_str.Get(TDI_CPU_FREQ).c_str();\n        map_str[TDI_GPU_TEMP].label = theApp.m_main_wnd_data.disp_str.Get(TDI_GPU_TEMP).c_str();\n        map_str[TDI_HDD_TEMP].label = theApp.m_main_wnd_data.disp_str.Get(TDI_HDD_TEMP).c_str();\n        map_str[TDI_MAIN_BOARD_TEMP].label = theApp.m_main_wnd_data.disp_str.Get(TDI_MAIN_BOARD_TEMP).c_str();\n    }\n\n    //上传/下载\n    CString in_speed = CCommon::DataSizeToString(theApp.m_in_speed, theApp.m_main_wnd_data);\n    CString out_speed = CCommon::DataSizeToString(theApp.m_out_speed, theApp.m_main_wnd_data);\n    CString total_speed = CCommon::DataSizeToString(theApp.m_in_speed + theApp.m_out_speed, theApp.m_main_wnd_data);\n    if (!theApp.m_main_wnd_data.hide_unit || theApp.m_main_wnd_data.speed_unit == SpeedUnit::AUTO)\n    {\n        in_speed += _T(\"/s\");\n        out_speed += _T(\"/s\");\n        total_speed += _T(\"/s\");\n    }\n    map_str[TDI_UP].value = out_speed.GetString();\n    map_str[TDI_DOWN].value = in_speed.GetString();\n    map_str[TDI_TOTAL_SPEED].value = total_speed.GetString();\n\n    if (theApp.m_main_wnd_data.swap_up_down) //交换上传和下载位置\n    {\n        std::swap(map_str[TDI_UP], map_str[TDI_DOWN]);\n    }\n\n    //CPU/内存/显卡利用率\n    map_str[TDI_CPU].value = CCommon::UsageToString(theApp.m_cpu_usage, theApp.m_main_wnd_data);\n \n    map_str[TDI_CPU_FREQ].value = CCommon::FreqToString(theApp.m_cpu_freq, theApp.m_main_wnd_data);\n    CString str_memory_value;\n    if (theApp.m_main_wnd_data.memory_display == MemoryDisplay::MEMORY_USED)\n        str_memory_value = CCommon::DataSizeToString(static_cast<unsigned long long>(theApp.m_used_memory) * 1024, theApp.m_main_wnd_data.separate_value_unit_with_space);\n    else if (theApp.m_main_wnd_data.memory_display == MemoryDisplay::MEMORY_AVAILABLE)\n        str_memory_value = CCommon::DataSizeToString((static_cast<unsigned long long>(theApp.m_total_memory) - static_cast<unsigned long long>(theApp.m_used_memory)) * 1024, theApp.m_main_wnd_data.separate_value_unit_with_space);\n    else\n        str_memory_value = CCommon::UsageToString(theApp.m_memory_usage, theApp.m_main_wnd_data);\n    map_str[TDI_MEMORY].value = str_memory_value;\n    map_str[TDI_GPU_USAGE].value = CCommon::UsageToString(theApp.m_gpu_usage, theApp.m_main_wnd_data);\n    map_str[TDI_HDD_USAGE].value = CCommon::UsageToString(theApp.m_hdd_usage, theApp.m_main_wnd_data);\n\n    //温度\n    auto getTemperatureStr = [&](DisplayItem display_item, float temperature)\n    {\n        map_str[display_item].value = CCommon::TemperatureToString(temperature, theApp.m_main_wnd_data);\n    };\n    getTemperatureStr(TDI_CPU_TEMP, theApp.m_cpu_temperature);\n    getTemperatureStr(TDI_GPU_TEMP, theApp.m_gpu_temperature);\n    getTemperatureStr(TDI_HDD_TEMP, theApp.m_hdd_temperature);\n    getTemperatureStr(TDI_MAIN_BOARD_TEMP, theApp.m_main_board_temperature);\n\n    //获取文本颜色\n    std::map<CommonDisplayItem, COLORREF> text_colors{};\n    if (theApp.m_main_wnd_data.specify_each_item_color)\n    {\n        text_colors = theApp.m_main_wnd_data.text_colors;\n    }\n    else if (!theApp.m_main_wnd_data.text_colors.empty())\n    {\n        for (const auto& item : theApp.m_plugins.AllDisplayItemsWithPlugins())\n        {\n            text_colors[item] = theApp.m_main_wnd_data.text_colors.begin()->second;\n        }\n    }\n\n    //绘制文本\n    draw.SetFont(&font);\n\n    //绘制文本\n    int index{};\n    for (auto iter = map_str.begin(); iter != map_str.end(); ++iter)\n    {\n        const auto& layout_item = layout.GetItem(iter->first);\n        if (layout_item.show)\n        {\n            //矩形区域\n            CRect rect(CPoint(layout_item.x, layout_item.y), CSize(layout_item.width, m_layout_info.text_height));\n\n            //文本颜色\n            COLORREF text_color = text_colors[iter->first];\n\n            //绘制文本\n            DrawSkinText(draw, map_str[iter->first], rect, text_color, layout_item.align);\n        }\n        index++;\n    }\n\n    //绘制插件项目\n    for (const auto& plugin_item : theApp.m_plugins.GetPluginItems())\n    {\n        const auto& layout_item = layout.GetItem(plugin_item);\n        if (layout_item.show)\n        {\n            //插件项目自绘\n            COLORREF cl{};\n            auto iter = text_colors.find(plugin_item);\n            if (iter != text_colors.end())\n                cl = iter->second;\n            else if (!text_colors.empty())\n                cl = text_colors.begin()->second;\n            if (plugin_item->IsCustomDraw())\n            {\n                int brightness{ (GetRValue(cl) + GetGValue(cl) + GetBValue(cl)) / 2 };\n                ITMPlugin* plugin = theApp.m_plugins.GetPluginByItem(plugin_item);\n                if (plugin != nullptr && plugin->GetAPIVersion() >= 2)\n                {\n                    plugin->OnExtenedInfo(ITMPlugin::EI_VALUE_TEXT_COLOR, std::to_wstring(cl).c_str());\n                    plugin->OnExtenedInfo(ITMPlugin::EI_DRAW_TASKBAR_WND, L\"0\");\n                }\n                draw.GetDC()->SetTextColor(cl);\n                plugin_item->DrawItem(draw.GetDC()->GetSafeHdc(), layout_item.x, layout_item.y, layout_item.width, m_layout_info.text_height, brightness >= 128);\n            }\n            else\n            {\n                //矩形区域\n                CRect rect(CPoint(layout_item.x, layout_item.y), CSize(layout_item.width, m_layout_info.text_height));\n\n                //绘制文本\n                DrawStr draw_str;\n                draw_str.label = theApp.m_main_wnd_data.disp_str.Get(plugin_item).c_str();\n                draw_str.value = plugin_item->GetItemValueText();\n                DrawSkinText(draw, draw_str, rect, cl, layout_item.align);\n            }\n        }\n    }\n}\n\nstring CSkinFile::GetDisplayItemXmlNodeName(DisplayItem display_item)\n{\n    switch (display_item)\n    {\n    case TDI_UP:\n        return \"up\";\n        break;\n    case TDI_DOWN:\n        return \"down\";\n        break;\n    case TDI_TOTAL_SPEED:\n        return \"total_speed\";\n        break;\n    case TDI_CPU:\n        return \"cpu\";\n        break;\n    case TDI_MEMORY:\n        return \"memory\";\n        break;\n    case TDI_GPU_USAGE:\n        return \"gpu\";\n        break;\n    case TDI_CPU_TEMP:\n        return \"cpu_temperature\";\n        break;\n    case TDI_GPU_TEMP:\n        return \"gpu_temperature\";\n        break;\n    case TDI_HDD_TEMP:\n        return \"hdd_temperature\";\n        break;\n    case TDI_MAIN_BOARD_TEMP:\n        return \"main_board_temperature\";\n        break;\n    case TDI_HDD_USAGE:\n        return \"hdd\";\n        break;\n    case TDI_CPU_FREQ:\n        return \"cpu_freq\";\n        break;\n    default:\n        return string();\n        break;\n    }\n}\n"
  },
  {
    "path": "TrafficMonitor/SkinFile.h",
    "content": "﻿#pragma once\n#include \"CommonData.h\"\n#include \"TinyXml2Helper.h\"\n#include \"DrawCommon.h\"\n\nclass CSkinFile\n{\npublic:\n    CSkinFile();\n    ~CSkinFile();\n\n\n    //从文件载入皮肤信息\n    void Load(const wstring& file_path);\n\n    //皮肤信息\n    struct SkinInfo\n    {\n        std::vector<COLORREF> text_color;   //文本颜色\n        bool specify_each_item_color{};     //是否指定每个项目的颜色\n        wstring skin_author;                //皮肤的作者\n        FontInfo font_info;                 //字体信息\n        DispStrings display_text;    //每一项的显示文本\n\n        COLORREF TextColor(size_t i) const\n        {\n            if (i >= 0 && i < text_color.size())\n                return text_color[i];\n            else if (!text_color.empty())\n                return text_color.front();\n            else\n                return 0;\n        }\n    };\n\n    //皮肤中每一项的布局信息\n    struct LayoutItem\n    {\n        int x{};            //X位置\n        int y{};            //Y位置\n        int width{};        //宽度\n        Alignment align{};  //对齐方式\n        bool show{};        //是否显示\n    };\n\n    //皮肤布局\n    struct Layout\n    {\n        int width{};        //宽度\n        int height{};       //高度\n        std::map<CommonDisplayItem, LayoutItem> layout_items; //每一项的布局信息\n        LayoutItem GetItem(CommonDisplayItem display_item) const\n        {\n            auto iter = layout_items.find(display_item);\n            if (iter != layout_items.end())\n                return iter->second;\n            return LayoutItem();\n        }\n    };\n\n    //皮肤布局信息\n    struct LayoutInfo\n    {\n        int text_height{};  //皮肤文本的高度\n        bool no_label{};    //是否不显示标签\n        Layout layout_l;    //“显示更多信息”时的布局\n        Layout layout_s;    //不“显示更多信息”时的布局\n    };\n\n    //皮肤预览图信息\n    struct PreviewInfo\n    {\n        struct Pos\n        {\n            int x{};\n            int y{};\n        };\n        int width{};    //预览图的宽度\n        int height{};   //预览图的高度\n        Pos l_pos;      //“显示更多信息”时的窗口在预览图中的位置\n        Pos s_pos;      //不“显示更多信息”时的窗口在预览图中的位置\n    };\n\n    const SkinInfo& GetSkinInfo() const { return m_skin_info; }\n    const LayoutInfo& GetLayoutInfo() const { return m_layout_info; }\n    const PreviewInfo& GetPreviewInfo() const { return m_preview_info; }\n\n    const CImage& GetBackgroundL() const { return m_background_l; }\n    const CImage& GetBackgroundS() const { return m_background_s; }\n\n    //绘制预览图\n    //pDC: 绘图的CDC\n    //rect: 绘图区域\n    void DrawPreview(CDC* pDC, CRect rect);\n\n    //绘制主界面\n    void DrawInfo(CDC* pDC, bool show_more_info, CFont& font);\n\n    static string GetDisplayItemXmlNodeName(DisplayItem display_item);\n\nprivate:\n    void LoadFromXml(const wstring& file_path);     //从xml文件读取皮肤数据\n    void LoadFromIni(const wstring& file_path);     //从ini文件读取皮肤数据（用于兼容旧版皮肤）\n\n    CSkinFile::Layout LayoutFromXmlNode(tinyxml2::XMLElement* ele);\n\n    struct DrawStr\n    {\n        CString label;\n        CString value;\n        CString GetStr() const\n        {\n            return label + value;\n        }\n    };\n\n    static void DrawSkinText(CDrawCommon drawer, DrawStr draw_str, CRect rect, COLORREF color, Alignment align);\n\nprivate:\n    SkinInfo m_skin_info;\n    LayoutInfo m_layout_info;\n    PreviewInfo m_preview_info;\n    std::map<std::string, std::string> m_plugin_map;  //插件名称与xml节点名称的映射关系。key是xml节点名称，value是插件ID\n\n    CFont m_font;\n    CImage m_background_s;\n    CImage m_background_l;\n\n};\n"
  },
  {
    "path": "TrafficMonitor/StaticEx.cpp",
    "content": "#include \"stdafx.h\"\n#include \"StaticEx.h\"\n\n\nCStaticEx::CStaticEx()\n{\n}\n\n\nCStaticEx::~CStaticEx()\n{\n}\n\nvoid CStaticEx::SetWindowTextEx(LPCTSTR lpszString, Alignment align)\n{\n\tm_text = lpszString;\n\tm_align = align;\n\tm_color_text = true;\n\tInvalidate();\n}\n\nvoid CStaticEx::SetTextColor(COLORREF textColor)\n{\n\tm_text_color = textColor;\n\tInvalidate();\n}\n\nvoid CStaticEx::SetBackColor(COLORREF back_color)\n{\n\tm_back_color = back_color;\n\tm_draw_background_color = true;\n}\n\nCString CStaticEx::GetString() const\n{\n\treturn m_text;\n}\n\n\nLRESULT CStaticEx::DefWindowProc(UINT message, WPARAM wParam, LPARAM lParam)\n{\n\t// TODO: ڴרô/û\n\tif (message == WM_SETTEXT)\n\t{\n\t\tCRect rect;\n\t\tCDC* pDC = GetDC();\n\t\tGetClientRect(rect);\n\t\tDrawThemeParentBackground(m_hWnd, pDC->GetSafeHdc(), &rect);\n\t\tReleaseDC(pDC);\n\t}\n\treturn CStatic::DefWindowProc(message, wParam, lParam);\n}\nBEGIN_MESSAGE_MAP(CStaticEx, CStatic)\n\tON_WM_PAINT()\nEND_MESSAGE_MAP()\n\n\n\nvoid CStaticEx::OnPaint()\n{\n\tCPaintDC dc(this); // device context for painting\n\t\t\t\t\t   // TODO: ڴ˴Ϣ\n\t\t\t\t\t   // ΪͼϢ CStatic::OnPaint()\n\tdc.SetTextColor(m_text_color);\n\tdc.SetBkMode(TRANSPARENT);\n\tdc.SelectObject(this->GetFont());\n\tCRect rect;\n\tthis->GetClientRect(&rect);\n\tif (m_draw_background_color)\n\t\tdc.FillSolidRect(rect, m_back_color);\n\telse\n\t\tDrawThemeParentBackground(m_hWnd, dc.GetSafeHdc(), &rect);\t//ػؼԽص\n\tCSize text_size = dc.GetTextExtent(m_text);\n\tUINT format{ DT_VCENTER | DT_SINGLELINE | DT_NOPREFIX };\t\t//CDC::DrawText()ıʽ\n\tif (text_size.cx > rect.Width())\t\t//ıȳ˾Ŀȣ˾ʱ\n\t{\n\t\tif (m_align == Alignment::RIGHT)\n\t\t\tformat |= DT_RIGHT;\n\t}\n\telse\n\t{\n\t\tswitch (m_align)\n\t\t{\n\t\tcase Alignment::RIGHT: format |= DT_RIGHT; break;\n\t\tcase Alignment::CENTER: format |= DT_CENTER; break;\n\t\t}\n\t}\n\tdc.DrawText(m_text, rect, format);\n}\n\n\nvoid CStaticEx::PreSubclassWindow()\n{\n\t// TODO: ڴרô/û\n\tGetWindowText(m_text);\n\n\tCStatic::PreSubclassWindow();\n}\n"
  },
  {
    "path": "TrafficMonitor/StaticEx.h",
    "content": "/*\nCStatic࣬ɫıؼ\nSetTextColorıɫ\nҪʱSetWindowTextExÿؼı\n*/\n#pragma once\n#include \"afxwin.h\"\n#include \"CommonData.h\"\nclass CStaticEx :\n\tpublic CStatic\n{\npublic:\n\tCStaticEx();\n\t~CStaticEx();\n\n\t//Staticؼɫıʱ\npublic:\n\tvoid SetWindowTextEx(LPCTSTR lpszString, Alignment align = Alignment::LEFT);\t//ΪؼɫıҪSetTextColorʹã\n\tvoid SetTextColor(COLORREF textColor);\t\t//ÿؼıɫ\n\tvoid SetBackColor(COLORREF back_color);\t\t//ÿؼɫ\n\tCString GetString() const;\t\t\t//ȡؼı\n\nprotected:\n\tbool m_color_text{ false };\n\tCOLORREF m_text_color;\t//ؼɫ\n\tCOLORREF m_back_color;\n\tCString m_text;\t\t\t//ؼϵı\n\tAlignment m_align{};\t\t//ıĶ뷽ʽ\n\tbool m_draw_background_color{ false };\t//ǷҪΪؼ䱳ɫ\n\nprotected:\n\tvirtual LRESULT DefWindowProc(UINT message, WPARAM wParam, LPARAM lParam);\npublic:\n\tDECLARE_MESSAGE_MAP()\n\tafx_msg void OnPaint();\n\tvirtual void PreSubclassWindow();\n};\n\n"
  },
  {
    "path": "TrafficMonitor/TabDlg.cpp",
    "content": "﻿#include \"stdafx.h\"\n#include \"TabDlg.h\"\n#include \"TrafficMonitor.h\"\n\nIMPLEMENT_DYNAMIC(CTabDlg, CDialogEx)\n\nCTabDlg::CTabDlg(UINT nIDTemplate, CWnd * pParent) : CDialogEx(nIDTemplate, pParent)\n{\n\tm_pParent = pParent;\n}\n\n\nCTabDlg::~CTabDlg()\n{\n}\n\n\nBOOL CTabDlg::PreTranslateMessage(MSG* pMsg)\n{\n\t// TODO: 在此添加专用代码和/或调用基类\n    //由于这是tab标签中的子对话框，因此用户按回车或ESC后不应该响应当前对话框的IDOK或IDCANCEL，\n    //而应该响应tab标签的父窗口的IDOK或IDCANCEL命令\n    if (pMsg->message == WM_KEYDOWN && pMsg->wParam == VK_ESCAPE)\n\t{\n\t\tif (m_pParent != nullptr)\n\t\t\tm_pParent->SendMessage(WM_COMMAND, IDCANCEL);\n\t\treturn TRUE;\n\t}\n\tif (pMsg->message == WM_KEYDOWN && pMsg->wParam == VK_RETURN)\n\t{\n\t\tif (m_pParent != nullptr)\n\t\t\tm_pParent->SendMessage(WM_COMMAND, IDOK);\n\t\treturn TRUE;\n\t}\n\n\treturn CDialogEx::PreTranslateMessage(pMsg);\n}\n\n\nBOOL CTabDlg::OnInitDialog()\n{\n\tCDialogEx::OnInitDialog();\n\n\t// TODO:  在此添加额外的初始化\n\t//将窗口背景设置成白色\n    if (theApp.m_win_version.IsWindows11OrLater())\n\t    SetBackgroundColor(RGB(249, 249, 249));\n    else\n\t    SetBackgroundColor(RGB(255, 255, 255));\n\n\treturn TRUE;  // return TRUE unless you set the focus to a control\n\t\t\t\t  // 异常: OCX 属性页应返回 FALSE\n}\n\nCWnd* CTabDlg::GetParentWindow()\n{\n    CWnd* pParent = GetParent();\n    if (pParent != nullptr)\n    {\n        return pParent->GetParent();\n    }\n    return nullptr;\n}\n\nvoid CTabDlg::SetScrollbarInfo(int nPage, int nMax)\n{\n\t//初始化滚动条\n\tSCROLLINFO scrollinfo;\n\tGetScrollInfo(SB_VERT, &scrollinfo, SIF_ALL);\n\tscrollinfo.nPage = nPage;    //设置滑块大小\n\tscrollinfo.nMin = 0;\n\tscrollinfo.nMax = nMax;     //设置滚动条的最大位置\n\tif (scrollinfo.nMax < 0)\n\t\tscrollinfo.nMax = 0;\n\tscrollinfo.nPos = scrollinfo.nMin;\n\tSetScrollInfo(SB_VERT, &scrollinfo, SIF_ALL);\n\n    m_scroll_enable = true;\n}\n\nvoid CTabDlg::ResetScroll()\n{\n\tif (m_scroll_enable)\n\t{\n\t\tm_last_pos = 0;\n\t\tSCROLLINFO scrollinfo;\n\t\tGetScrollInfo(SB_VERT, &scrollinfo, SIF_ALL);\n\t\tint step = scrollinfo.nPos - scrollinfo.nMin;\n\t\tscrollinfo.nPos = scrollinfo.nMin;\n\t\tSetScrollInfo(SB_VERT, &scrollinfo, SIF_ALL);\n\t\tScrollWindow(0, step);\n\t}\n}\n\nvoid CTabDlg::ScrollWindowSimple(int step)\n{\n\tSCROLLINFO scrollinfo;\n\tGetScrollInfo(SB_VERT, &scrollinfo, SIF_ALL);\n\tscrollinfo.nPos -= step;\n\tif (scrollinfo.nPos < scrollinfo.nMin)\n\t{\n\t\tstep = scrollinfo.nPos + step - scrollinfo.nMin;\t\t//如果向上滚动一个距离后小于滚动条的最小位置了，则修正step的值，使窗口滚动到最上方\n\t\tscrollinfo.nPos = scrollinfo.nMin;\n\t}\n\tif (scrollinfo.nPos + scrollinfo.nPage > scrollinfo.nMax)  //此处一定要注意加上滑块的长度，再作判断\n\t{\n\t\tstep -= (scrollinfo.nMax - (scrollinfo.nPos + scrollinfo.nPage));\t\t//如果向上滚动一个距离后大于滚动条的最大位置了，则修正step的值，使窗口滚动到最下方\n\t\tif (step > 0)\n\t\t\tstep = 0;\n\t\tscrollinfo.nPos = scrollinfo.nMax;\n\t}\n\tSetScrollInfo(SB_VERT, &scrollinfo, SIF_ALL);\n\tScrollWindow(0, step);\n\n}\n\nvoid CTabDlg::EnableDlgCtrl(UINT id, bool enable)\n{\n    CWnd* pCtrl = GetDlgItem(id);\n    if (pCtrl != nullptr)\n        pCtrl->EnableWindow(enable);\n}\n\nvoid CTabDlg::ShowDlgCtrl(UINT id, bool show)\n{\n    CWnd* pCtrl = GetDlgItem(id);\n    if (pCtrl != nullptr)\n        pCtrl->ShowWindow(show);\n}\n\nBEGIN_MESSAGE_MAP(CTabDlg, CDialogEx)\n\tON_WM_VSCROLL()\n\tON_WM_MOUSEWHEEL()\n    ON_WM_SIZE()\nEND_MESSAGE_MAP()\n\n\nvoid CTabDlg::OnVScroll(UINT nSBCode, UINT nPos, CScrollBar* pScrollBar)\n{\n\t// TODO: 在此添加消息处理程序代码和/或调用默认值\n\t//参考资料：https://www.cnblogs.com/ranjiewen/p/6013922.html\n\n\tif (m_scroll_enable)\n\t{\n\t\tSCROLLINFO scrollinfo;\n\t\tGetScrollInfo(SB_VERT, &scrollinfo, SIF_ALL);\n\t\tint unit = 1;\n\t\tint step = theApp.DPI(16);\n\t\tswitch (nSBCode)\n\t\t{\n\t\tcase SB_LINEUP:          //Scroll one line up\n\t\t\tScrollWindowSimple(unit * step);\n\t\t\tbreak;\n\t\tcase SB_LINEDOWN:           //Scroll one line down\n\t\t\tScrollWindowSimple(-unit * step);\n\t\t\tbreak;\n\t\tcase SB_PAGEUP:            //Scroll one page up.\n\t\t\tScrollWindowSimple(unit * step * 5);\n\t\t\tbreak;\n\t\tcase SB_PAGEDOWN:        //Scroll one page down        \n\t\t\tScrollWindowSimple(-unit * step * 5);\n\t\t\tbreak;\n\t\tcase SB_ENDSCROLL:      //End scroll     \n\t\t\tbreak;\n\t\tcase SB_THUMBPOSITION:  //Scroll to the absolute position. The current position is provided in nPos\n\t\t\tbreak;\n\t\tcase SB_THUMBTRACK:                  //Drag scroll box to specified position. The current position is provided in nPos\n        {\n            int y_amount = (m_last_pos - nPos)*unit;\n            ScrollWindow(0, y_amount);\n            scrollinfo.nPos = nPos;\n            SetScrollInfo(SB_VERT, &scrollinfo, SIF_ALL);\n        }\n\t\t\tbreak;\n\t\t}\n        m_last_pos = scrollinfo.nPos;\n\t}\n\tCDialogEx::OnVScroll(nSBCode, nPos, pScrollBar);\n}\n\n\nBOOL CTabDlg::OnMouseWheel(UINT nFlags, short zDelta, CPoint pt)\n{\n\t// TODO: 在此添加消息处理程序代码和/或调用默认值\n\tif (m_scroll_enable)\n\t{\n\t\tint step = theApp.DPI(64);\n\t\tif (zDelta > 0)\n\t\t{\n\t\t\tScrollWindowSimple(step);\n\t\t}\n\t\tif (zDelta < 0)\n\t\t{\n\t\t\tScrollWindowSimple(-step);\n\t\t}\n        SCROLLINFO scrollinfo;\n        GetScrollInfo(SB_VERT, &scrollinfo, SIF_ALL);\n        m_last_pos = scrollinfo.nPos;\n    }\n\n\treturn CDialogEx::OnMouseWheel(nFlags, zDelta, pt);\n}\n\n\nvoid CTabDlg::OnOK()\n{\n    // TODO: 在此添加专用代码和/或调用基类\n\n    CDialogEx::OnOK();\n}\n\n\nbool CTabDlg::IsScrollBarVisible()\n{\n    SCROLLINFO scrollinfo;\n    GetScrollInfo(SB_VERT, &scrollinfo, SIF_ALL);\n    bool visible = scrollinfo.nPage < scrollinfo.nMax;\n    return visible;\n}\n\nvoid CTabDlg::OnSize(UINT nType, int cx, int cy)\n{\n    CDialogEx::OnSize(nType, cx, cy);\n\n    // TODO: 在此处添加消息处理程序代码\n    SetControlMouseWheelEnable(!IsScrollBarVisible());      //如果显示了滚动条，则禁止控件响应鼠标滚轮，此时鼠标滚轮用于滚动窗口\n}\n"
  },
  {
    "path": "TrafficMonitor/TabDlg.h",
    "content": "﻿//这是用于Tab标签中的子对话框类\n#pragma once\nclass CTabDlg : public CDialogEx\n{\n\tDECLARE_DYNAMIC(CTabDlg)\npublic:\n\tCTabDlg(UINT nIDTemplate, CWnd *pParent = NULL);\n\t~CTabDlg();\n\tvirtual BOOL PreTranslateMessage(MSG* pMsg);\n\tvirtual BOOL OnInitDialog();\n    CWnd* GetParentWindow();\n\tvoid SetScrollbarInfo(int nPage, int nMax);\n    void ResetScroll();\n    virtual void OnTabEntered() {}      //当标签切换到当前窗口时被调用\n    virtual void OnOK();\n\tvoid SetScrollEnable(bool enable) { m_scroll_enable = enable; }\n    virtual void SetControlMouseWheelEnable(bool enable) {}     //在派生类中重写此函数以设置控件是否允许响应鼠标滚轮\n\n    bool IsScrollBarVisible();    //滚动条是否可见\n\nprotected:\n\tvoid ScrollWindowSimple(int step);\n\n    void EnableDlgCtrl(UINT id, bool enable);\n    void ShowDlgCtrl(UINT id, bool show);\n\nprotected:\n\tCWnd* m_pParent;\n\tbool m_scroll_enable{ false };\n    int m_last_pos{};\n\npublic:\n\tDECLARE_MESSAGE_MAP()\n\tafx_msg void OnVScroll(UINT nSBCode, UINT nPos, CScrollBar* pScrollBar);\n\tafx_msg BOOL OnMouseWheel(UINT nFlags, short zDelta, CPoint pt);\n    afx_msg void OnSize(UINT nType, int cx, int cy);\n};\n\n"
  },
  {
    "path": "TrafficMonitor/TaskBarDlg.cpp",
    "content": "﻿// TaskBarDlg.cpp : 实现文件\n//\n\n#include \"stdafx.h\"\n#include \"TrafficMonitor.h\"\n#include \"TaskBarDlg.h\"\n#include \"afxdialogex.h\"\n#include \"TrafficMonitorDlg.h\"\n#include \"WindowsSettingHelper.h\"\n\n\n// CTaskBarDlg 对话框\n\nIMPLEMENT_DYNAMIC(CTaskBarDlg, CDialogEx)\n\nCTaskBarDlg::CTaskBarDlg(CWnd* pParent /*=NULL*/)\n    : CDialogEx(IDD_TASK_BAR_DIALOG, pParent)\n{\n\n}\n\nCTaskBarDlg::~CTaskBarDlg()\n{\n    for (auto iter = m_map_history_data.begin(); iter != m_map_history_data.end(); ++iter)\n    {\n        iter->second.clear();\n    }\n}\n\nvoid CTaskBarDlg::DoDataExchange(CDataExchange* pDX)\n{\n    CDialogEx::DoDataExchange(pDX);\n}\n\n\nBEGIN_MESSAGE_MAP(CTaskBarDlg, CDialogEx)\n    ON_WM_RBUTTONUP()\n    ON_WM_INITMENU()\n    ON_WM_MOUSEMOVE()\n    ON_WM_LBUTTONDBLCLK()\n    ON_WM_TIMER()\n    ON_WM_PAINT()\n    ON_WM_CLOSE()\n    ON_WM_LBUTTONUP()\n    ON_MESSAGE(WM_EXITMENULOOP, &CTaskBarDlg::OnExitmenuloop)\nEND_MESSAGE_MAP()\n\n\n// CTaskBarDlg 消息处理程序\n\n\nvoid CTaskBarDlg::ShowInfo(CDC* pDC)\n{\n    if (this->GetSafeHwnd() == NULL || pDC == nullptr || !IsWindow(this->GetSafeHwnd())) return;\n\n    if (m_rect.IsRectEmpty() || m_rect.IsRectNull()) return;\n    CRect draw_rect{ m_rect };      //绘图的矩形区域\n    draw_rect.MoveToXY(0, 0);\n    //设置缓冲的DC\n    CDrawDoubleBuffer draw_double_buffer(pDC, draw_rect);\n    //绘图\n    CDrawCommon draw;\n    draw.Create(draw_double_buffer.GetMemDC(), nullptr);\n    draw.FillRect(draw_rect, theApp.m_taskbar_data.back_color);       //填充背景色\n    draw.SetFont(&m_font);\n    draw.SetBackColor(theApp.m_taskbar_data.back_color);\n\n    //计算各部分的位置\n    int index = 0;\n    CRect item_rect{};\n    int item_count = static_cast<int>(m_item_widths.size());  //要显示的项目数量\n    auto last_iter = m_item_widths.begin();\n    for (auto iter = m_item_widths.begin(); iter != m_item_widths.end(); ++iter)\n    {\n        auto last_item_width = last_iter->item_width;\n        //任务栏在桌面顶部或底部\n        if (IsTasksbarOnTopOrBottom())\n        {\n            if (theApp.m_taskbar_data.horizontal_arrange)   //水平排列\n            {\n                if (index > 0)\n                    item_rect.MoveToX(item_rect.right + theApp.DPI(theApp.m_taskbar_data.item_space));\n                item_rect.right = item_rect.left + iter->item_width.TotalWidth();\n                item_rect.bottom = item_rect.top + m_window_height;\n                if (iter->is_plugin)\n                    DrawPluginItem(draw, iter->plugin_item, item_rect, iter->item_width.label_width);\n                else\n                    DrawDisplayItem(draw, iter->item_type, item_rect, iter->item_width.label_width);\n            }\n            else        //非水平排列时，每两个一组显示\n            {\n                //在index为奇数时同时绘制两个项目\n                if (index % 2 == 1)\n                {\n                    CRect item_rect_up;     //上面一个项目的矩形区域\n                    if (index > 0)\n                        item_rect_up.MoveToXY(item_rect.right + theApp.DPI(theApp.m_taskbar_data.item_space), 0);\n                    item_rect.left = item_rect_up.left;\n                    item_rect.top = (m_window_height - TASKBAR_WND_HEIGHT / 2);\n                    //确定窗口大小\n                    item_rect_up.bottom = item_rect.top - 1;\n                    item_rect.bottom = m_window_height;\n                    int width = max(iter->item_width.TotalWidth(), last_item_width.TotalWidth());\n                    item_rect.right = item_rect.left + width;\n                    item_rect_up.right = item_rect_up.left + width;\n                    //绘制信息\n                    if (last_iter->is_plugin)\n                        DrawPluginItem(draw, last_iter->plugin_item, item_rect_up, last_item_width.label_width);\n                    else\n                        DrawDisplayItem(draw, last_iter->item_type, item_rect_up, last_item_width.label_width);\n\n                    if (iter->is_plugin)\n                        DrawPluginItem(draw, iter->plugin_item, item_rect, iter->item_width.label_width);\n                    else\n                        DrawDisplayItem(draw, iter->item_type, item_rect, iter->item_width.label_width);\n                }\n                //要绘制的项目为奇数时绘制最后一个\n                else if (item_count % 2 == 1 && index == item_count - 1)\n                {\n                    item_rect.MoveToXY(item_rect.right + theApp.DPI(theApp.m_taskbar_data.item_space), 0);\n                    item_rect.bottom = TASKBAR_WND_HEIGHT;\n                    item_rect.right = item_rect.left + iter->item_width.MaxWidth();\n                    if (iter->is_plugin)\n                        DrawPluginItem(draw, iter->plugin_item, item_rect, iter->item_width.label_width, true);\n                    else\n                        DrawDisplayItem(draw, iter->item_type, item_rect, iter->item_width.label_width, true);\n                }\n            }\n        }\n        //任务栏在桌面两侧\n        else\n        {\n            if (index > 0)\n                item_rect.MoveToXY(0, item_rect.bottom + theApp.DPI(theApp.m_taskbar_data.item_space));\n            item_rect.bottom = item_rect.top + TASKBAR_WND_HEIGHT / 2;\n            item_rect.right = item_rect.left + min(m_window_width, m_rcMin.Width() - theApp.DPI(theApp.m_taskbar_data.item_space));\n            if (iter->is_plugin)\n                DrawPluginItem(draw, iter->plugin_item, item_rect, iter->item_width.label_width);\n            else\n                DrawDisplayItem(draw, iter->item_type, item_rect, iter->item_width.label_width);\n        }\n\n        index++;\n        last_iter = iter;\n    }\n}\n\nvoid CTaskBarDlg::DrawDisplayItem(CDrawCommon& drawer, DisplayItem type, CRect rect, int label_width, bool vertical)\n{\n    m_item_rects[type] = rect;\n    //设置要绘制的文本颜色\n    COLORREF label_color{};\n    COLORREF text_color{};\n    if (theApp.m_taskbar_data.specify_each_item_color)\n    {\n        label_color = theApp.m_taskbar_data.text_colors[type].label;\n        text_color = theApp.m_taskbar_data.text_colors[type].value;\n    }\n    else if (!theApp.m_taskbar_data.text_colors.empty())\n    {\n        label_color = theApp.m_taskbar_data.text_colors.begin()->second.label;\n        text_color = theApp.m_taskbar_data.text_colors.begin()->second.label;\n    }\n\n    //设置标签和数值的矩形区域\n    CRect rect_label{ rect };\n    CRect rect_value{ rect };\n    if (!vertical)\n    {\n        rect_label.right = rect_label.left + label_width;\n        rect_value.left = rect_label.right;\n    }\n    else if (label_width != 0)\n    {\n        rect_label.bottom = rect_label.top + (rect_label.Height() / 2);\n        rect_value.top = rect_label.bottom;\n    }\n\n    // 绘制状态条\n    if (type == TDI_CPU || type == TDI_MEMORY || type == TDI_GPU_USAGE || type == TDI_CPU_TEMP\n        || type == TDI_GPU_TEMP || type == TDI_HDD_TEMP || type == TDI_MAIN_BOARD_TEMP || type == TDI_HDD_USAGE\n        || type == TDI_UP || type == TDI_DOWN || type == TDI_TOTAL_SPEED/* ||type==TDI_CPU_FREQ*/)\n    {\n        int figure_value{};\n        switch (type)\n        {\n        case TDI_CPU:\n            figure_value = theApp.m_cpu_usage;\n            break;\n        case TDI_MEMORY:\n            figure_value = theApp.m_memory_usage;\n            break;\n        case TDI_GPU_USAGE:\n            figure_value = theApp.m_gpu_usage;\n            break;\n        case TDI_CPU_TEMP:\n            figure_value = theApp.m_cpu_temperature;\n            break;\n        case TDI_GPU_TEMP:\n            figure_value = theApp.m_gpu_temperature;\n            break;\n        case TDI_HDD_TEMP:\n            figure_value = theApp.m_hdd_temperature;\n            break;\n        case TDI_MAIN_BOARD_TEMP:\n            figure_value = theApp.m_main_board_temperature;\n            break;\n        case TDI_HDD_USAGE:\n            figure_value = theApp.m_hdd_usage;\n            break;\n        //case TDI_CPU_FREQ:\n        //    figure_value = theApp.m_cpu_freq;\n        //    break;\n        case TDI_UP:\n            figure_value = CalculateNetspeedPercent(theApp.m_out_speed);\n            break;\n        case TDI_DOWN:\n            figure_value = CalculateNetspeedPercent(theApp.m_in_speed);\n            break;\n        case TDI_TOTAL_SPEED:\n            figure_value = CalculateNetspeedPercent(theApp.m_in_speed + theApp.m_out_speed);\n            break;\n        default:\n            break;\n        }\n\n        if ((type != TDI_UP && type != TDI_DOWN && type != TDI_TOTAL_SPEED) && theApp.m_taskbar_data.show_status_bar\n            || (type == TDI_UP || type == TDI_DOWN || type == TDI_TOTAL_SPEED) && theApp.m_taskbar_data.show_netspeed_figure)\n        {\n            if (theApp.m_taskbar_data.cm_graph_type)\n            {\n                AddHisToList(type, figure_value);\n                TryDrawGraph(drawer, rect, type);\n            }\n            else\n            {\n                TryDrawStatusBar(drawer, rect, figure_value);\n            }\n        }\n    }\n\n    //绘制标签\n    if (label_width > 0)\n    {\n        wstring str_label = theApp.m_taskbar_data.disp_str.Get(type);\n        //if (theApp.m_taskbar_data.swap_up_down)\n        //{\n        //    if (type == TDI_UP)\n        //        str_label = theApp.m_taskbar_data.disp_str.Get(TDI_DOWN);\n        //    else if (type == TDI_DOWN)\n        //        str_label = theApp.m_taskbar_data.disp_str.Get(TDI_UP);\n        //}\n        drawer.DrawWindowText(rect_label, str_label.c_str(), label_color, (vertical ? Alignment::CENTER : Alignment::LEFT));\n    }\n\n    //绘制数值\n    CString str_value;\n    Alignment value_alignment{ theApp.m_taskbar_data.value_right_align ? Alignment::RIGHT : Alignment::LEFT };      //数值的对齐方式\n    if (vertical)\n        value_alignment = Alignment::CENTER;\n    //绘制上传或下载速度\n    if (type == TDI_UP || type == TDI_DOWN || type == TDI_TOTAL_SPEED)\n    {\n        CString format_str;\n        if (theApp.m_taskbar_data.hide_unit && theApp.m_taskbar_data.speed_unit != SpeedUnit::AUTO)\n            format_str = _T(\"%s\");\n        else\n            format_str = _T(\"%s/s\");\n        CString str_in_speed = CCommon::DataSizeToString(theApp.m_in_speed, theApp.m_taskbar_data);\n        CString str_out_speed = CCommon::DataSizeToString(theApp.m_out_speed, theApp.m_taskbar_data);\n        CString str_total_speed = CCommon::DataSizeToString(theApp.m_in_speed + theApp.m_out_speed, theApp.m_taskbar_data);\n        //if (theApp.m_taskbar_data.swap_up_down)\n        //    std::swap(str_in_speed, str_out_speed);\n        if (type == TDI_UP)\n        {\n            str_value.Format(format_str, str_out_speed.GetString());\n        }\n        else if (type == TDI_DOWN)\n        {\n            str_value.Format(format_str, str_in_speed.GetString());\n        }\n        else\n        {\n            str_value.Format(format_str, str_total_speed.GetString());\n        }\n    }\n\n    //当内存显示为已使用内存或可用内存时\n    if (type == TDI_MEMORY && (theApp.m_taskbar_data.memory_display == MemoryDisplay::MEMORY_USED || theApp.m_taskbar_data.memory_display == MemoryDisplay::MEMORY_AVAILABLE))\n    {\n        if (theApp.m_taskbar_data.memory_display == MemoryDisplay::MEMORY_USED)\n            str_value = CCommon::DataSizeToString(static_cast<unsigned long long>(theApp.m_used_memory) * 1024, theApp.m_taskbar_data.separate_value_unit_with_space);\n        else\n            str_value = CCommon::DataSizeToString((static_cast<unsigned long long>(theApp.m_total_memory) - static_cast<unsigned long long>(theApp.m_used_memory)) * 1024, theApp.m_taskbar_data.separate_value_unit_with_space);\n    }\n    //绘制CPU或内存利用率\n    else if (type == TDI_CPU || type == TDI_MEMORY || type == TDI_GPU_USAGE || type == TDI_HDD_USAGE)\n    {\n        int usage{};\n        switch (type)\n        {\n        case TDI_CPU:\n            usage = theApp.m_cpu_usage;\n            break;\n        case TDI_MEMORY:\n            usage = theApp.m_memory_usage;\n            break;\n        case TDI_GPU_USAGE:\n            usage = theApp.m_gpu_usage;\n            break;\n        case TDI_HDD_USAGE:\n            usage = theApp.m_hdd_usage;\n            break;\n        default:\n            break;\n        }\n        str_value = CCommon::UsageToString(usage, theApp.m_taskbar_data);\n\n        //如果CPU或内存利用率达到100%，会导致显示不全，此时将绘图区域向右扩展一些\n        int text_width = m_pDC->GetTextExtent(str_value).cx;\n        if (usage >= 100 && rect_value.Width() < text_width)\n            rect_value.right = rect_value.left + text_width;\n    }\n\n    //绘制温度\n    else if (type == TDI_CPU_TEMP || type == TDI_GPU_TEMP || type == TDI_HDD_TEMP || type == TDI_MAIN_BOARD_TEMP)\n    {\n        int temperature{};\n        switch (type)\n        {\n        case TDI_CPU_TEMP:\n            temperature = theApp.m_cpu_temperature;\n            break;\n        case TDI_GPU_TEMP:\n            temperature = theApp.m_gpu_temperature;\n            break;\n        case TDI_HDD_TEMP:\n            temperature = theApp.m_hdd_temperature;\n            break;\n        case TDI_MAIN_BOARD_TEMP:\n            temperature = theApp.m_main_board_temperature;\n            break;\n        default:\n            break;\n        }\n        str_value = CCommon::TemperatureToString(temperature, theApp.m_taskbar_data);\n    }\n    else if (type == TDI_CPU_FREQ) {\n        str_value = CCommon::FreqToString(theApp.m_cpu_freq, theApp.m_taskbar_data);\n    }\n\n    drawer.DrawWindowText(rect_value, str_value, text_color, value_alignment);\n}\n\nvoid CTaskBarDlg::DrawPluginItem(CDrawCommon& drawer, IPluginItem* item, CRect rect, int label_width, bool vertical)\n{\n    if (item == nullptr)\n        return;\n    m_item_rects[item] = rect;\n    //设置要绘制的文本颜色\n    COLORREF label_text_color{};\n    COLORREF value_text_color{};\n    if (theApp.m_taskbar_data.specify_each_item_color)\n    {\n        label_text_color = theApp.m_taskbar_data.text_colors[item].label;\n        value_text_color = theApp.m_taskbar_data.text_colors[item].value;\n    }\n    else if (!theApp.m_taskbar_data.text_colors.empty())\n    {\n        label_text_color = theApp.m_taskbar_data.text_colors.begin()->second.label;\n        value_text_color = theApp.m_taskbar_data.text_colors.begin()->second.label;\n    }\n\n    if (item->IsCustomDraw())\n    {\n        //根据背景色的亮度判断深色还是浅色模式\n        const COLORREF& bk{ theApp.m_taskbar_data.back_color };\n        int background_brightness{ (GetRValue(bk) + GetGValue(bk) + GetBValue(bk)) / 3 };\n        //由插件自绘\n        ITMPlugin* plugin = theApp.m_plugins.GetPluginByItem(item);\n        if (plugin != nullptr && plugin->GetAPIVersion() >= 2)\n        {\n            plugin->OnExtenedInfo(ITMPlugin::EI_LABEL_TEXT_COLOR, std::to_wstring(label_text_color).c_str());\n            plugin->OnExtenedInfo(ITMPlugin::EI_VALUE_TEXT_COLOR, std::to_wstring(value_text_color).c_str());\n            plugin->OnExtenedInfo(ITMPlugin::EI_DRAW_TASKBAR_WND, L\"1\");\n        }\n        drawer.GetDC()->SetTextColor(value_text_color);\n        item->DrawItem(drawer.GetDC()->GetSafeHdc(), rect.left, rect.top, rect.Width(), rect.Height(), background_brightness < 128);\n    }\n    else\n    {\n        CRect rect_label, rect_value;\n        rect_label = rect_value = rect;\n        if (label_width > 0)\n        {\n            if (!vertical)\n            {\n                rect_label = rect_value = rect;\n                rect_label.right = rect_label.left + label_width;\n                rect_value.left = rect_label.right;\n            }\n            else\n            {\n                rect_label.bottom = rect_label.top + rect.Height() / 2;\n                rect_value.top = rect_label.bottom;\n            }\n        }\n        //画标签\n        CString lable_text = theApp.m_taskbar_data.disp_str.Get(item).c_str();\n        lable_text += L' ';\n        drawer.DrawWindowText(rect_label, lable_text, label_text_color, (vertical ? Alignment::CENTER : Alignment::LEFT));\n        //画数值\n        Alignment value_alignment{ theApp.m_taskbar_data.value_right_align ? Alignment::RIGHT : Alignment::LEFT };      //数值的对齐方式\n        if (vertical)\n            value_alignment = Alignment::CENTER;\n        drawer.DrawWindowText(rect_value, item->GetItemValueText(), value_text_color, value_alignment);\n    }\n}\n\nvoid CTaskBarDlg::MoveWindow(CRect rect)\n{\n    if (IsWindow(GetSafeHwnd()))\n    {\n        ::MoveWindow(GetSafeHwnd(), rect.left, rect.top, rect.Width(), rect.Height(), TRUE);\n    }\n}\n\nvoid CTaskBarDlg::TryDrawStatusBar(CDrawCommon& drawer, const CRect& rect_bar, int usage_percent)\n{\n    CSize fill_size = CSize(rect_bar.Width() * usage_percent / 100, rect_bar.Height());\n    CRect rect_fill(rect_bar.TopLeft(), fill_size);\n    if (theApp.m_taskbar_data.show_graph_dashed_box)\n        drawer.DrawRectOutLine(rect_bar, theApp.m_taskbar_data.status_bar_color, 1, true);\n    drawer.FillRect(rect_fill, theApp.m_taskbar_data.status_bar_color);\n}\n\nbool CTaskBarDlg::AdjustWindowPos()\n{\n    if (this->GetSafeHwnd() == NULL || !IsWindow(this->GetSafeHwnd()))\n        return false;\n    ::GetWindowRect(m_hMin, m_rcMin); //获得最小化窗口的区域\n    ::GetWindowRect(m_hBar, m_rcBar); //获得二级容器的区域\n    static bool last_taskbar_on_top_or_bottom;\n    CheckTaskbarOnTopOrBottom();\n    if (m_taskbar_on_top_or_bottom != last_taskbar_on_top_or_bottom)\n    {\n        CalculateWindowSize();\n        last_taskbar_on_top_or_bottom = m_taskbar_on_top_or_bottom;\n    }\n\n    if (m_taskbar_on_top_or_bottom)     //当任务栏在桌面顶部或底部时\n    {\n        //设置窗口大小\n        m_rect.right = m_rect.left + m_window_width;\n        m_rect.bottom = m_rect.top + m_window_height;\n        if (m_rcMin.Width() != m_min_bar_width)   //如果最小化窗口的宽度改变了，重新设置任务栏窗口的位置\n        {\n            m_rcMinOri = m_rcMin;\n            m_left_space = m_rcMin.left - m_rcBar.left;\n            m_min_bar_width = m_rcMin.Width() - m_rect.Width(); //保存最小化窗口宽度\n            if (!theApp.m_taskbar_data.tbar_wnd_on_left)\n            {\n                if (theApp.m_is_windows11_taskbar)\n                {\n                    if (!theApp.m_taskbar_data.tbar_wnd_snap)\n                        m_rect.MoveToX(m_rcBar.Width() - m_rect.Width() + 2);\n                    else\n                        m_rect.MoveToX(m_rcMin.right + 2);\n                }\n                else\n                {\n                    ::MoveWindow(m_hMin, m_left_space, 0, m_rcMin.Width() - m_rect.Width(), m_rcMin.Height(), TRUE);    //设置最小化窗口的位置\n                    m_rect.MoveToX(m_left_space + m_rcMin.Width() - m_rect.Width() + 2);\n                }\n            }\n            else\n            {\n                if (theApp.m_is_windows11_taskbar)\n                {\n                    if (CWindowsSettingHelper::IsTaskbarCenterAlign())\n                    {\n                        if (theApp.m_taskbar_data.tbar_wnd_snap)\n                        {\n                            int taskbar_btn_num{ 1 };      //Win11任务栏“运行中的程序”左侧4个按钮（开始、搜索、任务视图、聊天）有几个显示。（“开始”按钮总是显示）\n                            if (CWindowsSettingHelper::IsTaskbarSearchBtnShown())\n                                taskbar_btn_num++;\n                            if (CWindowsSettingHelper::IsTaskbarTaskViewBtnShown())\n                                taskbar_btn_num++;\n                            if (CWindowsSettingHelper::IsTaskbarChartBtnShown())\n                                taskbar_btn_num++;\n\n                            m_rect.MoveToX(m_rcMin.left - m_rect.Width() - 2 - theApp.DPI(44) * taskbar_btn_num);   //每个按钮44像素\n                        }\n                        else\n                        {\n                            if (CWindowsSettingHelper::IsTaskbarWidgetsBtnShown())\n                                m_rect.MoveToX(2 + theApp.DPI(theApp.m_cfg_data.taskbar_left_space_win11));\n                            else\n                                m_rect.MoveToX(2);\n                        }\n                    }\n                    else\n                    {\n                        m_rect.MoveToX(2);\n                    }\n                }\n                else\n                {\n                    ::MoveWindow(m_hMin, m_left_space + m_rect.Width(), 0, m_rcMin.Width() - m_rect.Width(), m_rcMin.Height(), TRUE);\n                    m_rect.MoveToX(m_left_space);\n                }\n            }\n            m_rect.MoveToY((m_rcBar.Height() - m_rect.Height()) / 2);\n            if (theApp.m_taskbar_data.horizontal_arrange && theApp.m_win_version.IsWindows7())\n                m_rect.MoveToY(m_rect.top + theApp.DPI(1));\n            MoveWindow(m_rect);\n        }\n    }\n    else        //当任务栏在屏幕在左侧或右侧时\n    {\n        //设置窗口大小\n        if (m_rcMin.Height() != m_min_bar_height) //如果最小化窗口的高度改变了，重新设置任务栏窗口的位置\n        {\n            m_rcMinOri = m_rcMin;\n            m_top_space = m_rcMin.top - m_rcBar.top;\n            m_min_bar_height = m_rcMin.Height() - m_rect.Height();  //保存最小化窗口高度\n            if (!theApp.m_taskbar_data.tbar_wnd_on_left)\n            {\n                ::MoveWindow(m_hMin, 0, m_top_space, m_rcMin.Width(), m_rcMin.Height() - m_rect.Height(), TRUE);    //设置最小化窗口的位置\n                m_rect.MoveToY(m_top_space + m_rcMin.Height() - m_rect.Height() + 2);\n            }\n            else\n            {\n                ::MoveWindow(m_hMin, 0, m_top_space + m_rect.Height(), m_rcMin.Width(), m_rcMin.Height() - m_rect.Height(), TRUE);  //设置最小化窗口的位置\n                m_rect.MoveToY(m_top_space);\n            }\n            m_rect.MoveToX((m_rcMin.Width() - m_window_width) / 2);\n            if (m_rect.left < theApp.DPI(2))\n                m_rect.MoveToX(theApp.DPI(2));\n            MoveWindow(m_rect);\n        }\n    }\n\n    CRect rect{ m_rect };\n    //如果窗口没有被成功嵌入到任务栏，窗口移动到了基于屏幕左上角的绝对位置，则修正窗口的位置\n    if (m_connot_insert_to_task_bar)\n    {\n        rect.MoveToXY(rect.left + m_rcBar.left, rect.top + m_rcBar.top);\n        this->MoveWindow(rect);\n    }\n\n    if (m_connot_insert_to_task_bar && ::GetForegroundWindow() == m_hTaskbar)   //在窗口无法嵌入任务栏时，如果焦点设置在了任务栏上，则让窗口置顶\n    {\n        SetWindowPos(&wndTopMost, 0, 0, 0, 0, SWP_NOSIZE | SWP_NOMOVE);         //设置置顶\n    }\n    return true;\n}\n\nvoid CTaskBarDlg::ApplyWindowTransparentColor()\n{\n#ifndef COMPILE_FOR_WINXP\n    if (theApp.m_is_windows11_taskbar)      //Windows11下背景色不使用纯黑色，以解决深色模式下右键菜单无法弹出的问题\n    {\n        if (theApp.m_taskbar_data.transparent_color == 0 && theApp.m_taskbar_data.back_color == 0)\n        {\n            theApp.m_taskbar_data.transparent_color = 1;\n            theApp.m_taskbar_data.back_color = 1;\n        }\n    }\n    if ((theApp.m_taskbar_data.transparent_color != 0) && theApp.m_taksbar_transparent_color_enable)\n    {\n        SetWindowLong(m_hWnd, GWL_EXSTYLE, GetWindowLong(m_hWnd, GWL_EXSTYLE) | WS_EX_LAYERED);\n        SetLayeredWindowAttributes(theApp.m_taskbar_data.transparent_color, 0, LWA_COLORKEY);\n    }\n    else\n    {\n        SetWindowLong(m_hWnd, GWL_EXSTYLE, GetWindowLong(m_hWnd, GWL_EXSTYLE) & ~WS_EX_LAYERED);\n    }\n#endif // !COMPILE_FOR_WINXP\n}\n\n\nvoid CTaskBarDlg::CheckTaskbarOnTopOrBottom()\n{\n    CRect rect;\n    CRect rcMin;\n    CRect rcBar;\n    if (m_hTaskbar != 0)\n    {\n        ::GetWindowRect(m_hMin, rcMin); //获得最小化窗口的区域\n        ::GetWindowRect(m_hBar, rcBar); //获得二级容器的区域\n        if (m_left_space == 0)\n            m_left_space = rcMin.left - rcBar.left;\n        if (m_top_space == 0)\n            m_top_space = rcMin.top - rcBar.top;\n\n        ::GetWindowRect(m_hTaskbar, rect);          //获取任务栏的矩形区域\n        m_taskbar_on_top_or_bottom = (rect.Width() >= rect.Height());     //如果任务栏的宽度大于高度，则任务在屏幕的顶部或底部\n    }\n    else\n    {\n        m_taskbar_on_top_or_bottom = true;\n    }\n}\n\nCString CTaskBarDlg::GetMouseTipsInfo()\n{\n    CString tip_info;\n    CString temp;\n    temp.Format(_T(\"%s: %s\\r\\n (%s: %s/%s: %s)\"), CCommon::LoadText(IDS_TRAFFIC_USED_TODAY),\n        CCommon::KBytesToString((theApp.m_today_up_traffic + theApp.m_today_down_traffic) / 1024u),\n        CCommon::LoadText(IDS_UPLOAD), CCommon::KBytesToString(theApp.m_today_up_traffic / 1024u),\n        CCommon::LoadText(IDS_DOWNLOAD), CCommon::KBytesToString(theApp.m_today_down_traffic / 1024u)\n    );\n    tip_info += temp;\n    if (!IsItemShow(TDI_UP))\n    {\n        temp.Format(_T(\"\\r\\n%s: %s/s\"), CCommon::LoadText(IDS_UPLOAD),\n            CCommon::DataSizeToString(theApp.m_out_speed, theApp.m_main_wnd_data));\n        tip_info += temp;\n    }\n    if (!IsItemShow(TDI_DOWN))\n    {\n        temp.Format(_T(\"\\r\\n%s: %s/s\"), CCommon::LoadText(IDS_DOWNLOAD),\n            CCommon::DataSizeToString(theApp.m_in_speed, theApp.m_main_wnd_data));\n        tip_info += temp;\n    }\n    if (!IsItemShow(TDI_CPU))\n    {\n        temp.Format(_T(\"\\r\\n%s: %d %%\"), CCommon::LoadText(IDS_CPU_USAGE), theApp.m_cpu_usage);\n        tip_info += temp;\n    }\n    if (!IsShowCpuMemory())\n    {\n        temp.Format(_T(\"\\r\\n%s: %s/%s\"),\n            CCommon::LoadText(IDS_MEMORY_USAGE),\n            CCommon::KBytesToString(theApp.m_used_memory), CCommon::KBytesToString(theApp.m_total_memory));\n    }\n    if (!IsItemShow(TDI_MEMORY))\n    {\n        temp.Format(_T(\"\\r\\n%s: %s/%s (%d %%)\"), CCommon::LoadText(IDS_MEMORY_USAGE),\n            CCommon::KBytesToString(theApp.m_used_memory),\n            CCommon::KBytesToString(theApp.m_total_memory), theApp.m_memory_usage);\n        tip_info += temp;\n    }\n    else\n    {\n        temp.Format(_T(\"\\r\\n%s: %s/%s\"), CCommon::LoadText(IDS_MEMORY_USAGE),\n            CCommon::KBytesToString(theApp.m_used_memory),\n            CCommon::KBytesToString(theApp.m_total_memory));\n        tip_info += temp;\n    }\n#ifndef WITHOUT_TEMPERATURE\n    CTrafficMonitorDlg* pMainWnd = dynamic_cast<CTrafficMonitorDlg*>(theApp.m_pMainWnd);\n    if (pMainWnd->IsTemperatureNeeded())\n    {\n        if (!IsItemShow(TDI_GPU_USAGE) && theApp.m_gpu_usage >= 0)\n        {\n            temp.Format(_T(\"\\r\\n%s: %d %%\"), CCommon::LoadText(IDS_GPU_USAGE), theApp.m_gpu_usage);\n            tip_info += temp;\n        }\n        if (!IsItemShow(TDI_CPU_TEMP) && theApp.m_cpu_temperature > 0)\n        {\n            temp.Format(_T(\"\\r\\n%s: %s\"), CCommon::LoadText(IDS_CPU_TEMPERATURE), CCommon::TemperatureToString(theApp.m_cpu_temperature, theApp.m_taskbar_data));\n            tip_info += temp;\n        }\n        if (!IsItemShow(TDI_CPU_FREQ) && theApp.m_cpu_freq > 0)\n        {\n            temp.Format(_T(\"\\r\\n%s: %s\"), CCommon::LoadText(IDS_CPU_FREQ), CCommon::FreqToString(theApp.m_cpu_freq, theApp.m_taskbar_data));\n            tip_info += temp;\n        }\n        if (!IsItemShow(TDI_GPU_TEMP) && theApp.m_gpu_temperature > 0)\n        {\n            temp.Format(_T(\"\\r\\n%s: %s\"), CCommon::LoadText(IDS_GPU_TEMPERATURE), CCommon::TemperatureToString(theApp.m_gpu_temperature, theApp.m_taskbar_data));\n            tip_info += temp;\n        }\n        if (!IsItemShow(TDI_HDD_TEMP) && theApp.m_hdd_temperature > 0)\n        {\n            temp.Format(_T(\"\\r\\n%s: %s\"), CCommon::LoadText(IDS_HDD_TEMPERATURE), CCommon::TemperatureToString(theApp.m_hdd_temperature, theApp.m_taskbar_data));\n            tip_info += temp;\n        }\n        if (!IsItemShow(TDI_MAIN_BOARD_TEMP) && theApp.m_main_board_temperature > 0)\n        {\n            temp.Format(_T(\"\\r\\n%s: %s\"), CCommon::LoadText(IDS_MAINBOARD_TEMPERATURE), CCommon::TemperatureToString(theApp.m_main_board_temperature, theApp.m_taskbar_data));\n            tip_info += temp;\n        }\n        if (!IsItemShow(TDI_HDD_USAGE) && theApp.m_hdd_usage >= 0)\n        {\n            temp.Format(_T(\"\\r\\n%s: %d %%\"), CCommon::LoadText(IDS_HDD_USAGE), theApp.m_hdd_usage);\n            tip_info += temp;\n        }\n    }\n#endif\n\n    //添加插件项目的鼠标提示\n    tip_info += theApp.GetPlauginTooltipInfo().c_str();\n\n    return tip_info;\n}\n\nvoid CTaskBarDlg::SetTextFont()\n{\n    //如果m_font已经关联了一个字体资源对象，则释放它\n    if (m_font.m_hObject)\n    {\n        m_font.DeleteObject();\n    }\n    //创建新的字体\n    theApp.m_taskbar_data.font.Create(m_font, theApp.GetDpi());\n}\n\nvoid CTaskBarDlg::ApplySettings()\n{\n    SetTextFont();\n    CalculateWindowSize();\n}\n\nvoid CTaskBarDlg::CalculateWindowSize()\n{\n    bool horizontal_arrange = theApp.m_taskbar_data.horizontal_arrange && m_taskbar_on_top_or_bottom;\n    if (theApp.m_taskbar_data.m_tbar_display_item == 0)\n        theApp.m_taskbar_data.m_tbar_display_item |= TDI_UP;        //至少显示一项\n\n    m_item_widths.clear();\n    //显示项目的宽度\n    std::map<CommonDisplayItem, ItemWidth> item_widths;\n\n    m_pDC->SelectObject(&m_font);\n    //计算标签宽度\n    //const auto& item_map = theApp.m_taskbar_data.disp_str.GetAllItems();\n    for (auto iter = theApp.m_plugins.AllDisplayItemsWithPlugins().begin(); iter != theApp.m_plugins.AllDisplayItemsWithPlugins().end(); ++iter)\n    {\n        if (iter->is_plugin)\n        {\n            auto plugin = iter->plugin_item;\n            if (plugin != nullptr)\n            {\n                int& label_width{ item_widths[*iter].label_width };\n                if (plugin->IsCustomDraw())\n                {\n                    label_width = 0;\n                }\n                else\n                {\n                    CString lable_text = theApp.m_taskbar_data.disp_str.Get(plugin).c_str();\n                    if (!lable_text.IsEmpty())\n                        lable_text += L' ';\n                    label_width = m_pDC->GetTextExtent(lable_text).cx;\n                }\n            }\n        }\n        else\n        {\n            item_widths[*iter].label_width = m_pDC->GetTextExtent(theApp.m_taskbar_data.disp_str.Get(*iter).c_str()).cx;\n        }\n    }\n\n    //计算数值部分宽度\n\n    //计算显示上传下载部分所需要的宽度\n    CString sample_str;\n    int value_width{};\n    wstring digits(theApp.m_taskbar_data.digits_number, L'8');      //根据数据位数生成指定个数的“8”\n    bool hide_unit{ theApp.m_taskbar_data.hide_unit && theApp.m_taskbar_data.speed_unit != SpeedUnit::AUTO };\n    if (theApp.m_taskbar_data.speed_short_mode)\n    {\n        if (hide_unit)\n            sample_str.Format(_T(\"%s.\"), digits.c_str());\n        else\n            sample_str.Format(_T(\"%s.M/s\"), digits.c_str());\n    }\n    else\n    {\n        if (hide_unit)\n            sample_str.Format(_T(\"%s.8\"), digits.c_str());\n        else\n            sample_str.Format(_T(\"%s.8MB/s\"), digits.c_str());\n    }\n    if (!hide_unit && theApp.m_taskbar_data.separate_value_unit_with_space)\n        sample_str += _T(' ');\n    if (theApp.m_taskbar_data.speed_short_mode && !theApp.m_taskbar_data.unit_byte && !theApp.m_taskbar_data.hide_unit)\n        sample_str += _T('b');\n    value_width = m_pDC->GetTextExtent(sample_str).cx;      //计算使用当前字体显示文本需要的宽度值\n    item_widths[TDI_UP].value_width = value_width;\n    item_widths[TDI_DOWN].value_width = value_width;\n    item_widths[TDI_TOTAL_SPEED].value_width = value_width;\n\n    //计算显示CPU、内存部分所需要的宽度\n    CString str;\n    if (theApp.m_taskbar_data.hide_percent)\n    {\n        str = _T(\"99\");\n    }\n    else if (theApp.m_taskbar_data.separate_value_unit_with_space)\n    {\n        str = _T(\"99 %\");\n    }\n    else\n    {\n        str = _T(\"99%\");\n    }\n    value_width = m_pDC->GetTextExtent(str).cx;\n    //内存显示的宽度\n    int memory_width{ value_width };\n    if (theApp.m_taskbar_data.memory_display == MemoryDisplay::MEMORY_USED || theApp.m_taskbar_data.memory_display == MemoryDisplay::MEMORY_AVAILABLE)\n    {\n        if (theApp.m_taskbar_data.separate_value_unit_with_space)\n            str = _T(\"19.99 GB\");\n        else\n            str = _T(\"19.99GB\");\n        memory_width = m_pDC->GetTextExtent(str).cx;\n    }\n    item_widths[TDI_CPU].value_width = value_width;\n    item_widths[TDI_MEMORY].value_width = memory_width;\n    item_widths[TDI_GPU_USAGE].value_width = value_width;\n    item_widths[TDI_HDD_USAGE].value_width = value_width;\n\n    item_widths[TDI_CPU_FREQ].value_width = m_pDC->GetTextExtent(_T(\"1.00 GHz\")).cx;\n\n    //计算温度显示的宽度\n    if (theApp.m_taskbar_data.separate_value_unit_with_space)\n        str = _T(\"99 °C\");\n    else\n        str = _T(\"99°C\");\n    value_width = m_pDC->GetTextExtent(str).cx;\n    value_width += theApp.DPI(2);\n    item_widths[TDI_CPU_TEMP].value_width = value_width;\n    item_widths[TDI_GPU_TEMP].value_width = value_width;\n    item_widths[TDI_HDD_TEMP].value_width = value_width;\n    item_widths[TDI_MAIN_BOARD_TEMP].value_width = value_width;\n\n    //计算插件项目的宽度\n    for (const auto& plugin : theApp.m_plugins.GetPluginItems())\n    {\n        int& value_width{ item_widths[plugin].value_width };\n        if (plugin != nullptr && theApp.m_taskbar_data.plugin_display_item.Contains(plugin->GetItemId()))\n        {\n            if (plugin->IsCustomDraw())\n            {\n                value_width = theApp.m_plugins.GetItemWidth(plugin, m_pDC);\n            }\n            else\n            {\n                value_width = m_pDC->GetTextExtent(plugin->GetItemValueSampleText()).cx;\n            }\n        }\n    }\n\n    auto item_order{ theApp.m_taskbar_data.item_order.GetAllDisplayItemsWithOrder() };\n    for (const auto& item : item_order)\n    {\n        if (theApp.IsTaksbarItemDisplayed(item))\n        {\n            ItemWidthInfo width_info = item;\n            width_info.item_width = item_widths[item];\n            m_item_widths.push_back(width_info);\n        }\n    }\n\n    int item_count = static_cast<int>(m_item_widths.size());\n\n    //计算窗口总宽度\n    if (IsTasksbarOnTopOrBottom())  //任务栏在桌面的顶部或底部时\n    {\n        m_window_width = 0;\n        if (theApp.m_taskbar_data.horizontal_arrange)   //水平排列时\n        {\n            for (auto iter = m_item_widths.begin(); iter != m_item_widths.end(); ++iter)\n            {\n                m_window_width += iter->item_width.TotalWidth();\n            }\n            m_window_width += theApp.DPI(theApp.m_taskbar_data.item_space) * item_count;   //加上每个标签间的空隙\n        }\n        else        //非水平排列时，每两个一组排列\n        {\n            int index = 0;\n            int width0;\n            for (auto iter = m_item_widths.begin(); iter != m_item_widths.end(); ++iter)\n            {\n                if (index % 2 == 0)\n                {\n                    width0 = iter->item_width.TotalWidth();\n                }\n                else\n                {\n                    m_window_width += max(width0, iter->item_width.TotalWidth());\n                }\n                if (item_count % 2 == 1 && index == item_count - 1) //项目数为奇数时加上最后一个的宽度\n                {\n                    m_window_width += iter->item_width.MaxWidth();\n                }\n\n                index++;\n            }\n\n            m_window_width += theApp.DPI(theApp.m_taskbar_data.item_space) * ((item_count + 1) / 2 + 1);   //加上每个标签间的空隙\n        }\n    }\n    else        //任务栏在桌面两侧时\n    {\n        m_window_width = 0;\n        //所有标签中最大的宽度即为窗口宽度\n        for (auto iter = m_item_widths.begin(); iter != m_item_widths.end(); ++iter)\n        {\n            if (m_window_width < iter->item_width.TotalWidth())\n                m_window_width = iter->item_width.TotalWidth();\n        }\n    }\n\n    //计算窗口高度\n    if (IsTasksbarOnTopOrBottom())\n    {\n        if (!horizontal_arrange)\n            m_window_height = TASKBAR_WND_HEIGHT;\n        else\n            m_window_height = TASKBAR_WND_HEIGHT * 2 / 3;\n    }\n    else\n    {\n        m_window_height = TASKBAR_WND_HEIGHT / 2 * item_count;\n        m_window_height += (theApp.DPI(theApp.m_taskbar_data.item_space) * item_count);   //加上每个标签间的空隙\n    }\n    m_rect.right = m_rect.left + m_window_width;\n    m_rect.bottom = m_rect.top + m_window_height;\n\n}\n\nvoid CTaskBarDlg::SetToolTipsTopMost()\n{\n    m_tool_tips.SetWindowPos(&wndTopMost, 0, 0, 0, 0, SWP_NOSIZE | SWP_NOMOVE);\n}\n\nvoid CTaskBarDlg::UpdateToolTips()\n{\n    if (theApp.m_taskbar_data.show_tool_tip && IsWindow(m_tool_tips.GetSafeHwnd()))\n    {\n        CString tip_info;\n        tip_info = GetMouseTipsInfo();\n        m_tool_tips.UpdateTipText(tip_info, this);\n    }\n}\n\nbool CTaskBarDlg::IsItemShow(DisplayItem item)\n{\n    return (theApp.m_taskbar_data.m_tbar_display_item & item);\n}\n\nbool CTaskBarDlg::IsShowCpuMemory()\n{\n    return ((theApp.m_taskbar_data.m_tbar_display_item & TDI_CPU) || (theApp.m_taskbar_data.m_tbar_display_item & TDI_MEMORY));\n}\n\nbool CTaskBarDlg::IsShowNetSpeed()\n{\n    return ((theApp.m_taskbar_data.m_tbar_display_item & TDI_UP) || (theApp.m_taskbar_data.m_tbar_display_item & TDI_DOWN));\n}\n\n\nBOOL CTaskBarDlg::OnInitDialog()\n{\n    CDialogEx::OnInitDialog();\n\n    // TODO:  在此添加额外的初始化\n    //设置隐藏任务栏图标\n    ModifyStyleEx(0, WS_EX_TOOLWINDOW);\n\n    m_pDC = GetDC();\n\n\n    //设置字体\n    SetTextFont();\n    m_pDC->SelectObject(&m_font);\n\n\n    m_hTaskbar = ::FindWindow(L\"Shell_TrayWnd\", NULL);      //寻找类名是Shell_TrayWnd的窗口句柄\n    m_hBar = ::FindWindowEx(m_hTaskbar, 0, L\"ReBarWindow32\", NULL); //寻找二级容器的句柄\n    m_hMin = ::FindWindowEx(m_hBar, 0, L\"MSTaskSwWClass\", NULL);    //寻找最小化窗口的句柄\n\n    //在“Shell_TrayWnd”的子窗口找到类名为“Windows.UI.Composition.DesktopWindowContentBridge”的窗口则认为是Windows11的任务栏\n    if (theApp.m_win_version.IsWindows11OrLater())\n    {\n        theApp.m_is_windows11_taskbar = (::FindWindowExW(m_hTaskbar, 0, L\"Windows.UI.Composition.DesktopWindowContentBridge\", NULL) != NULL);\n    }\n    //设置窗口透明色\n    ApplyWindowTransparentColor();\n\n    ::GetWindowRect(m_hMin, m_rcMin);   //获得最小化窗口的区域\n    ::GetWindowRect(m_hBar, m_rcBar);   //获得二级容器的区域\n    m_left_space = m_rcMin.left - m_rcBar.left;\n    m_top_space = m_rcMin.top - m_rcBar.top;\n\n    CheckTaskbarOnTopOrBottom();\n    CalculateWindowSize();\n    m_rect.SetRectEmpty();\n    m_rect.bottom = m_window_height;\n    m_rect.right = m_rect.left + m_window_width;\n\n    m_connot_insert_to_task_bar = !(::SetParent(this->m_hWnd, m_hBar)); //把程序窗口设置成任务栏的子窗口\n    m_error_code = GetLastError();\n    AdjustWindowPos();\n\n    SetBackgroundColor(theApp.m_taskbar_data.back_color);\n\n    //初始化鼠标提示\n    if (IsWindow(GetSafeHwnd()) && m_tool_tips.Create(this, TTS_ALWAYSTIP) && IsWindow(m_tool_tips.GetSafeHwnd()))\n    {\n        m_tool_tips.SetMaxTipWidth(600);\n        m_tool_tips.AddTool(this, _T(\"\"));\n        SetToolTipsTopMost();       //设置提示信息总是置顶\n    }\n\n    //SetTimer(TASKBAR_TIMER, 100, NULL);\n\n    return TRUE;  // return TRUE unless you set the focus to a control\n                  // 异常: OCX 属性页应返回 FALSE\n}\n\n\nvoid CTaskBarDlg::OnCancel()\n{\n    // TODO: 在此添加专用代码和/或调用基类\n    //SaveConfig();\n\n    //关闭所有以任务栏窗口为父窗口的窗口\n    for (const auto& item : CBaseDialog::AllUniqueHandels())\n    {\n        HWND parent = ::GetParent(item.second);\n        if (parent == GetSafeHwnd())\n        {\n            ::SendMessage(item.second, WM_COMMAND, IDCANCEL, 0);\n        }\n    }\n\n    DestroyWindow();\n    //程序关闭的时候，把最小化窗口的width恢复回去\n    CheckTaskbarOnTopOrBottom();\n    if (m_taskbar_on_top_or_bottom)\n        ::MoveWindow(m_hMin, m_left_space, 0, m_rcMinOri.Width(), m_rcMinOri.Height(), TRUE);\n    else\n\n        ::MoveWindow(m_hMin, 0, m_top_space, m_rcMinOri.Width(), m_rcMinOri.Height(), TRUE);\n\n    //CDialogEx::OnCancel();\n}\n\n\nvoid CTaskBarDlg::OnRButtonUp(UINT nFlags, CPoint point)\n{\n    // TODO: 在此添加消息处理程序代码和/或调用默认值\n    m_menu_popuped = true;\n    m_tool_tips.Pop();\n    if (CheckClickedItem(point) && m_clicked_item.is_plugin && m_clicked_item.plugin_item != nullptr)\n    {\n        ITMPlugin* plugin = theApp.m_plugins.GetPluginByItem(m_clicked_item.plugin_item);\n        if (plugin != nullptr && plugin->GetAPIVersion() >= 3)\n        {\n            if (m_clicked_item.plugin_item->OnMouseEvent(IPluginItem::MT_RCLICKED, point.x, point.y, (void*)GetSafeHwnd(), IPluginItem::MF_TASKBAR_WND) != 0)\n                return;\n        }\n    }\n\n    CPoint point1;  //定义一个用于确定光标位置的位置\n    GetCursorPos(&point1);  //获取当前光标的位置，以便使得菜单可以跟随光标\n    CMenu* pMenu = theApp.m_taskbar_menu.GetSubMenu(0);\n    if (pMenu != nullptr)\n        pMenu->TrackPopupMenu(TPM_LEFTALIGN | TPM_RIGHTBUTTON, point1.x, point1.y, this); //在指定位置显示弹出菜单\n    CDialogEx::OnRButtonUp(nFlags, point1);\n}\n\n\nvoid CTaskBarDlg::OnInitMenu(CMenu* pMenu)\n{\n    CDialogEx::OnInitMenu(pMenu);\n\n    // TODO: 在此处添加消息处理程序代码\n    pMenu->CheckMenuItem(ID_SHOW_MAIN_WND, MF_BYCOMMAND | (!theApp.m_cfg_data.m_hide_main_window ? MF_CHECKED : MF_UNCHECKED));\n\n    pMenu->EnableMenuItem(ID_SELECT_ALL_CONNECTION, MF_BYCOMMAND | (theApp.m_general_data.show_all_interface ? MF_GRAYED : MF_ENABLED));\n    pMenu->EnableMenuItem(ID_CHECK_UPDATE, MF_BYCOMMAND | (theApp.IsCheckingForUpdate() ? MF_GRAYED : MF_ENABLED));\n\n    //pMenu->SetDefaultItem(ID_NETWORK_INFO);\n    //设置默认菜单项\n    switch (theApp.m_taskbar_data.double_click_action)\n    {\n    case DoubleClickAction::CONNECTION_INFO:\n        pMenu->SetDefaultItem(ID_NETWORK_INFO);\n        break;\n    case DoubleClickAction::HISTORY_TRAFFIC:\n        pMenu->SetDefaultItem(ID_TRAFFIC_HISTORY);\n        break;\n    case DoubleClickAction::SHOW_MORE_INFO:\n        pMenu->SetDefaultItem(ID_SHOW_CPU_MEMORY2);\n        break;\n    case DoubleClickAction::OPTIONS:\n        pMenu->SetDefaultItem(ID_OPTIONS2);\n        break;\n    case DoubleClickAction::TASK_MANAGER:\n        pMenu->SetDefaultItem(ID_OPEN_TASK_MANAGER);\n        break;\n    default:\n        pMenu->SetDefaultItem(-1);\n        break;\n    }\n    ::SendMessage(theApp.m_pMainWnd->GetSafeHwnd(), WM_TASKBAR_MENU_POPED_UP, 0, 0);        //通知主窗口菜单已弹出\n}\n\n\nBOOL CTaskBarDlg::PreTranslateMessage(MSG* pMsg)\n{\n    // TODO: 在此添加专用代码和/或调用基类\n    //屏蔽按回车键和ESC键退出\n    if (pMsg->message == WM_KEYDOWN && pMsg->wParam == VK_ESCAPE) return TRUE;\n    if (pMsg->message == WM_KEYDOWN && pMsg->wParam == VK_RETURN) return TRUE;\n\n    if (theApp.m_taskbar_data.show_tool_tip && !m_menu_popuped && IsWindow(m_tool_tips.GetSafeHwnd()) && (pMsg->message == WM_LBUTTONDOWN ||\n        pMsg->message == WM_LBUTTONUP ||\n        pMsg->message == WM_MOUSEMOVE))\n    {\n        m_tool_tips.RelayEvent(pMsg);\n    }\n\n    return CDialogEx::PreTranslateMessage(pMsg);\n}\n\n\nvoid CTaskBarDlg::OnMouseMove(UINT nFlags, CPoint point)\n{\n    // TODO: 在此添加消息处理程序代码和/或调用默认值\n\n    CDialogEx::OnMouseMove(nFlags, point);\n}\n\n\nvoid CTaskBarDlg::OnLButtonDblClk(UINT nFlags, CPoint point)\n{\n    // TODO: 在此添加消息处理程序代码和/或调用默认值\n    if (CheckClickedItem(point) && m_clicked_item.is_plugin && m_clicked_item.plugin_item != nullptr)\n    {\n        ITMPlugin* plugin = theApp.m_plugins.GetPluginByItem(m_clicked_item.plugin_item);\n        if (plugin != nullptr && plugin->GetAPIVersion() >= 3)\n        {\n            if (m_clicked_item.plugin_item->OnMouseEvent(IPluginItem::MT_DBCLICKED, point.x, point.y, (void*)GetSafeHwnd(), IPluginItem::MF_TASKBAR_WND) != 0)\n                return;\n        }\n    }\n    switch (theApp.m_taskbar_data.double_click_action)\n    {\n    case DoubleClickAction::CONNECTION_INFO:\n        SendMessage(WM_COMMAND, ID_NETWORK_INFO);        //双击后弹出“连接详情”对话框\n        break;\n    case DoubleClickAction::HISTORY_TRAFFIC:\n        SendMessage(WM_COMMAND, ID_TRAFFIC_HISTORY);        //双击后弹出“历史流量统计”对话框\n        break;\n    case DoubleClickAction::SHOW_MORE_INFO:\n        PostMessage(WM_COMMAND, ID_SHOW_CPU_MEMORY2);       //切换显示CPU和内存利用率\n        break;\n    case DoubleClickAction::OPTIONS:\n        SendMessage(WM_COMMAND, ID_OPTIONS2);       //双击后弹出“选项设置”对话框\n        break;\n    case DoubleClickAction::TASK_MANAGER:\n        ShellExecuteW(NULL, _T(\"open\"), (theApp.m_system_dir + L\"\\\\Taskmgr.exe\").c_str(), NULL, NULL, SW_NORMAL);       //打开任务管理器\n        break;\n    case DoubleClickAction::SEPCIFIC_APP:\n        ShellExecuteW(NULL, _T(\"open\"), (theApp.m_taskbar_data.double_click_exe).c_str(), NULL, NULL, SW_NORMAL);   //打开指定程序，默认任务管理器\n        break;\n    default:\n        break;\n    }\n    //CDialogEx::OnLButtonDblClk(nFlags, point);\n}\n\n\nvoid CTaskBarDlg::OnTimer(UINT_PTR nIDEvent)\n{\n    // TODO: 在此添加消息处理程序代码和/或调用默认值\n    //if (nIDEvent == TASKBAR_TIMER)\n    //{\n    //  AdjustWindowPos();\n    //  //ShowInfo();\n    //  Invalidate(FALSE);\n    //}\n\n    CDialogEx::OnTimer(nIDEvent);\n}\n\n\nBOOL CTaskBarDlg::OnCommand(WPARAM wParam, LPARAM lParam)\n{\n    // TODO: 在此添加专用代码和/或调用基类\n    UINT uMsg = LOWORD(wParam);\n    if (uMsg == ID_SELECT_ALL_CONNECTION || uMsg == ID_SELETE_CONNECTION\n        || (uMsg > ID_SELECT_ALL_CONNECTION && uMsg <= ID_SELETE_CONNECTION_MAX))\n    {\n        ::SendMessage(theApp.m_pMainWnd->GetSafeHwnd(), WM_COMMAND, wParam, lParam);    //如果点击了“选择网络连接”子菜单项，将消息转发到主窗口\n        return TRUE;\n    }\n\n    //选择了“显示项目”中的插件项目\n    if (uMsg >= ID_SHOW_PLUGIN_ITEM_START && uMsg <= ID_SHOW_PLUGIN_ITEM_MAX)\n    {\n        IPluginItem* item = theApp.m_plugins.GetItemByIndex(uMsg - ID_SHOW_PLUGIN_ITEM_START);\n        if (item != nullptr)\n        {\n            bool displayed = theApp.m_taskbar_data.plugin_display_item.Contains(item->GetItemId());\n            theApp.m_taskbar_data.plugin_display_item.SetStrContained(item->GetItemId(), !displayed);\n            ::PostMessage(theApp.m_pMainWnd->GetSafeHwnd(), WM_REOPEN_TASKBAR_WND, 0, 0);\n        }\n    }\n\n    return CDialogEx::OnCommand(wParam, lParam);\n}\n\n\nvoid CTaskBarDlg::OnPaint()\n{\n    CPaintDC dc(this); // device context for painting\n                       // TODO: 在此处添加消息处理程序代码\n                       // 不为绘图消息调用 CDialogEx::OnPaint()\n    ShowInfo(&dc);\n}\n\nvoid CTaskBarDlg::AddHisToList(DisplayItem item_type, int current_usage_percent)\n{\n    int& data_count{ m_history_data_count[item_type] };\n    std::list<int>& list = m_map_history_data[item_type];\n    //将数累加到加链表的头部，直到添加的数据数量达到TASKBAR_GRAPH_STEP的倍数时计算平均数\n    if (data_count % TASKBAR_GRAPH_STEP == 0)\n    {\n        //计算前面累加的TASKBAR_GRAPH_STEP个数据的平均数\n        if (!list.empty())\n            list.front() /= TASKBAR_GRAPH_STEP;\n        //将新的数据添加到末尾\n        list.push_front(current_usage_percent);\n    }\n    else\n    {\n        //数累加到加链表的头部\n        list.front() += current_usage_percent;\n    }\n    size_t graph_max_length = m_item_rects[item_type].Width();\n    //判断是否超过最大长度，如果超过，将链表尾部数据移除\n    if (list.size() > graph_max_length)\n    {\n        list.pop_back();\n    }\n    data_count++;\n}\n\n\nint CTaskBarDlg::CalculateNetspeedPercent(unsigned __int64 net_speed)\n{\n    int percet = 0;\n    unsigned __int64 max_value{ theApp.m_taskbar_data.GetNetspeedFigureMaxValueInBytes() };\n\n    if (net_speed >= max_value)\n        percet = 100;\n    else if (max_value > 0)\n        percet = net_speed * 100 / max_value;\n    return percet;\n}\n\nbool CTaskBarDlg::CheckClickedItem(CPoint point)\n{\n    for (const auto& item : m_item_rects)\n    {\n        if (item.second.PtInRect(point))\n        {\n            m_clicked_item = item.first;\n            return true;\n        }\n    }\n    m_clicked_item = TDI_UP;\n    return false;\n}\n\nvoid CTaskBarDlg::TryDrawGraph(CDrawCommon& drawer, const CRect& value_rect, DisplayItem item_type)\n{\n    std::list<int>& list = m_map_history_data[item_type];\n    if (theApp.m_taskbar_data.show_graph_dashed_box)\n        drawer.DrawRectOutLine(value_rect, theApp.m_taskbar_data.status_bar_color, 1, true);\n    int i{ -1 };\n    for (const auto& item : list)\n    {\n        i++;\n        if (i == 0)     //不绘制链表头部的数据，因为在累加中，还未取平均数\n            continue;\n        if (i >= value_rect.Width())\n            break;\n        //从右往左画线\n        CPoint start_point = CPoint(value_rect.right - i, value_rect.bottom);\n        int height = item * value_rect.Height() / 100;\n        drawer.DrawLine(start_point, height, theApp.m_taskbar_data.status_bar_color);\n    }\n}\n\n\nvoid CTaskBarDlg::OnClose()\n{\n    // TODO: 在此添加消息处理程序代码和/或调用默认值\n    ::SendMessage(theApp.m_pMainWnd->GetSafeHwnd(), WM_TASKBAR_WND_CLOSED, 0, 0);\n\n    CDialogEx::OnClose();\n}\n\n\nvoid CTaskBarDlg::OnLButtonUp(UINT nFlags, CPoint point)\n{\n    // TODO: 在此添加消息处理程序代码和/或调用默认值\n    if (CheckClickedItem(point) && m_clicked_item.is_plugin && m_clicked_item.plugin_item != nullptr)\n    {\n        ITMPlugin* plugin = theApp.m_plugins.GetPluginByItem(m_clicked_item.plugin_item);\n        if (plugin != nullptr && plugin->GetAPIVersion() >= 3)\n        {\n            if (m_clicked_item.plugin_item->OnMouseEvent(IPluginItem::MT_LCLICKED, point.x, point.y, (void*)GetSafeHwnd(), IPluginItem::MF_TASKBAR_WND) != 0)\n                return;\n        }\n    }\n\n    CDialogEx::OnLButtonUp(nFlags, point);\n}\n\n\nafx_msg LRESULT CTaskBarDlg::OnExitmenuloop(WPARAM wParam, LPARAM lParam)\n{\n    m_menu_popuped = false;\n    return 0;\n}\n"
  },
  {
    "path": "TrafficMonitor/TaskBarDlg.h",
    "content": "﻿#pragma once\n#include \"Common.h\"\n#include \"afxwin.h\"\n#include \"DrawCommon.h\"\n#include \"IniHelper.h\"\n#include \"CommonData.h\"\n#include \"TaskbarItemOrderHelper.h\"\n#include <list>\n\n// CTaskBarDlg 对话框\n#define TASKBAR_WND_HEIGHT theApp.DPI(32)\t\t\t\t//任务栏窗口的高度\n#define WM_TASKBAR_MENU_POPED_UP (WM_USER + 1004)\t\t//定义任务栏窗口右键菜单弹出时发出的消息\n//#define TASKBAR_GRAPH_MAX_LEN 600\t\t\t\t\t\t//历史数据存储最大长度\n#define TASKBAR_GRAPH_STEP 5\t\t\t\t\t\t\t//几秒钟画一条线\n\nclass CTaskBarDlg : public CDialogEx\n{\n    DECLARE_DYNAMIC(CTaskBarDlg)\n\npublic:\n    CTaskBarDlg(CWnd* pParent = NULL);   // 标准构造函数\n    virtual ~CTaskBarDlg();\n\n    CToolTipCtrl m_tool_tips;\n\n    void ShowInfo(CDC* pDC); \t//将信息绘制到控件上\n    void TryDrawStatusBar(CDrawCommon& drawer, const CRect& rect_bar, int usage_percent); //绘制CPU/内存状态条\n\n    void TryDrawGraph(CDrawCommon& drawer, const CRect& value_rect, DisplayItem item_type);\t\t// 绘制CPU/内存动态图\n\n    bool AdjustWindowPos();\t//设置窗口在任务栏中的位置\n    void ApplyWindowTransparentColor();\n\n    // 对话框数据\n#ifdef AFX_DESIGN_TIME\n    enum { IDD = IDD_TASK_BAR_DIALOG };\n#endif\n\nprotected:\n    virtual void DoDataExchange(CDataExchange* pDX);    // DDX/DDV 支持\n\n    HWND m_hTaskbar;\t//任务栏窗口句柄\n    HWND m_hBar;\t\t//任务栏窗口二级容器的句柄\n    HWND m_hMin;\t\t//最小化窗口的句柄\n    CRect m_rcBar;\t\t//初始状态时任务栏窗口的矩形区域\n    CRect m_rcMin;\t\t//最小化窗口的矩形区域\n    CRect m_rcMinOri;   //初始状态时最小化窗口的矩形区域\n    CRect m_rect;\t\t//当前窗口的矩形区域\n    int m_window_width{};\n    int m_window_height{};\n\n    //任务栏各个部分的宽度\n    struct ItemWidth\n    {\n        int label_width{};      //标签部分宽度\n        int value_width{};      //数值部分宽度\n\n        int TotalWidth() const  //总宽度\n        {\n            return label_width + value_width;\n        }\n\n        int MaxWidth() const\t//最大宽度\n        {\n            return max(label_width, value_width);\n        }\n    };\n\n    struct ItemWidthInfo : public CommonDisplayItem\n    {\n        ItemWidth item_width;\n\n        ItemWidthInfo()\n        {}\n\n        ItemWidthInfo(const CommonDisplayItem& item)\n            : CommonDisplayItem(item)\n        {}\n    };\n\n    std::vector<ItemWidthInfo> m_item_widths;   //任务栏窗口每个部分的宽度\n    std::map<CommonDisplayItem, CRect> m_item_rects;    //任务栏窗口每个部分的矩形区域\n    CommonDisplayItem m_clicked_item;           //鼠标点击的任务栏项目\n\n    int m_min_bar_width;\t//最小化窗口缩小宽度后的宽度\n    int m_min_bar_height;\t//最小化窗口缩小高度后的高度（用于任务栏在屏幕左侧或右侧时）\n\n    std::map<DisplayItem, std::list<int>> m_map_history_data;  //保存各项数据历史数据的链表，链表保存按照时间顺序，越靠近头部数据越新\n    std::map<DisplayItem, int> m_history_data_count;            //统计添加到历史数据链表的次数\n\n    int m_left_space{};\t\t\t//最小化窗口和二级窗口窗口左侧的边距\n    int m_top_space{};\t\t\t//最小化窗口和二级窗口窗口顶部的边距（用于任务栏在屏幕左侧或右侧时）\n\n    bool m_connot_insert_to_task_bar{ false };\t//如果窗口无法嵌入任务栏，则为true\n    bool m_taskbar_on_top_or_bottom{ true };\t\t//如果任务栏在屏幕顶部或底部，则为ture\n    int m_error_code{};\n    bool m_menu_popuped{ false };               //指示当前是否有菜单处于弹出状态\n\n    CFont m_font;\t\t\t//字体\n\n    CDC* m_pDC{};\t\t//窗口的DC，用来计算窗口的宽度\n\n    void CheckTaskbarOnTopOrBottom();\t\t//检查任务栏是否在屏幕的顶部或底部，并将结果保存在m_taskbar_on_top_or_bottom中\n    CString GetMouseTipsInfo();\t\t//获取鼠标提示\n\n    void AddHisToList(DisplayItem item_type, int current_usage_percent);\t\t//将当前利用率数值添加进链表\n\n    int CalculateNetspeedPercent(unsigned __int64 net_speed);     //计算网速占网速占用图的最大值的百分比\n\n    //判断一个点在哪个显示项目的区域内，并保存到m_clicked_item。如果返回false，则该点不在任何一个项目的区域内，否则返回true\n    bool CheckClickedItem(CPoint point);\n\n    //绘制任务栏窗口中的一个显示项目\n    //  drawer: 绘图类的对象\n    //  type: 项目的类型\n    //  rect: 绘制矩形区域\n    //  label_width: 标签区域的宽度\n    //  vertical: 如果为true，则标签和数值上下显示\n    void DrawDisplayItem(CDrawCommon& drawer, DisplayItem type, CRect rect, int label_width, bool vertical = false);\n\n    //绘制任务栏窗口中的一个插件项目\n   //  drawer: 绘图类的对象\n   //  item: 插件显示项目的指针\n   //  rect: 绘制矩形区域\n   //  label_width: 标签区域的宽度\n   //  vertical: 如果为true，则标签和数值上下显示\n    void DrawPluginItem(CDrawCommon& drawer, IPluginItem* item, CRect rect, int label_width, bool vertical = false);\n\n    void MoveWindow(CRect rect);\n\npublic:\n    void SetTextFont();\n    void ApplySettings();\n    void CalculateWindowSize();\t\t//计算窗口每部分的大小，及各个部分的宽度。窗口大小保存到m_window_width和m_window_height中，各部分宽度保存到m_item_widths中\n\n    void SetToolTipsTopMost();\t\t\t//设置鼠标提示置顶\n    void UpdateToolTips();\n\n    bool GetCannotInsertToTaskBar() const { return m_connot_insert_to_task_bar; }\n    int GetErrorCode() const { return m_error_code; }\n    bool IsTasksbarOnTopOrBottom() { return m_taskbar_on_top_or_bottom; }\n\n    static bool IsItemShow(DisplayItem item);\n    static bool IsShowCpuMemory();\n    static bool IsShowNetSpeed();\n\n    CommonDisplayItem GetClickedItem() const { return m_clicked_item; }\n\n    DECLARE_MESSAGE_MAP()\n\npublic:\n    virtual BOOL OnInitDialog();\n    virtual void OnCancel();\n    afx_msg void OnRButtonUp(UINT nFlags, CPoint point);\n    //afx_msg void OnSetBackgroundColor();\n    //afx_msg void OnSetTextColor();\n    afx_msg void OnInitMenu(CMenu* pMenu);\n    virtual BOOL PreTranslateMessage(MSG* pMsg);\n    //afx_msg void OnSetFont();\n    afx_msg void OnMouseMove(UINT nFlags, CPoint point);\n    afx_msg void OnLButtonDblClk(UINT nFlags, CPoint point);\n    afx_msg void OnTimer(UINT_PTR nIDEvent);\n    virtual BOOL OnCommand(WPARAM wParam, LPARAM lParam);\n    afx_msg void OnPaint();\n    afx_msg void OnClose();\n    afx_msg void OnLButtonUp(UINT nFlags, CPoint point);\nprotected:\n    afx_msg LRESULT OnExitmenuloop(WPARAM wParam, LPARAM lParam);\n};\n"
  },
  {
    "path": "TrafficMonitor/TaskBarSettingsDlg.cpp",
    "content": "﻿// TaskBarSettingsDlg.cpp : 实现文件\n//\n\n#include \"stdafx.h\"\n#include \"TrafficMonitor.h\"\n#include \"TaskBarSettingsDlg.h\"\n#include \"afxdialogex.h\"\n#include \"CMFCColorDialogEx.h\"\n#include \"CAutoAdaptSettingsDlg.h\"\n#include \"DisplayTextSettingDlg.h\"\n#include \"SetItemOrderDlg.h\"\n#include \"WindowsSettingHelper.h\"\n\n// CTaskBarSettingsDlg 对话框\n\nIMPLEMENT_DYNAMIC(CTaskBarSettingsDlg, CTabDlg)\n\nCTaskBarSettingsDlg::CTaskBarSettingsDlg(CWnd* pParent /*=NULL*/)\n    : CTabDlg(IDD_TASKBAR_SETTINGS_DIALOG, pParent)\n{\n\n}\n\nCTaskBarSettingsDlg::~CTaskBarSettingsDlg()\n{\n}\n\nbool CTaskBarSettingsDlg::IsStyleModified()\n{\n    bool modified{};\n    modified |= (theApp.m_taskbar_data.text_colors != m_data.text_colors);\n    modified |= (theApp.m_taskbar_data.back_color != m_data.back_color);\n    modified |= (theApp.m_taskbar_data.transparent_color != m_data.transparent_color);\n    modified |= (theApp.m_taskbar_data.status_bar_color != m_data.status_bar_color);\n    modified |= (theApp.m_taskbar_data.specify_each_item_color != m_data.specify_each_item_color);\n    return modified && m_style_modified;\n}\n\nvoid CTaskBarSettingsDlg::DrawStaticColor()\n{\n    //CCommon::FillStaticColor(m_text_color_static, m_data.text_color);\n    //CCommon::FillStaticColor(m_back_color_static, m_data.back_color);\n    if (m_data.specify_each_item_color)\n    {\n        int color_num{};\n#ifdef WITHOUT_TEMPERATURE\n        color_num = 8;\n#else\n        color_num = 16;\n#endif\n        int i{};\n        m_text_color_static.SetColorNum(color_num);\n        for (const auto& item : m_data.text_colors)\n        {\n            m_text_color_static.SetFillColor(i, item.second.label);\n            m_text_color_static.SetFillColor(i + 1, item.second.value);\n            i += 2;\n        }\n        m_text_color_static.Invalidate();\n    }\n    else if (!m_data.text_colors.empty())\n    {\n        m_text_color_static.SetFillColor(m_data.text_colors.begin()->second.label);\n    }\n    m_back_color_static.SetFillColor(m_data.back_color);\n    //m_trans_color_static.SetFillColor(m_data.transparent_color);\n    m_status_bar_color_static.SetFillColor(m_data.status_bar_color);\n}\n\nvoid CTaskBarSettingsDlg::IniUnitCombo()\n{\n    m_unit_combo.ResetContent();\n    m_unit_combo.AddString(CCommon::LoadText(IDS_AUTO));\n    if (m_data.unit_byte)\n    {\n        m_unit_combo.AddString(CCommon::LoadText(IDS_FIXED_AS, _T(\" KB/s\")));\n        m_unit_combo.AddString(CCommon::LoadText(IDS_FIXED_AS, _T(\" MB/s\")));\n    }\n    else\n    {\n        m_unit_combo.AddString(CCommon::LoadText(IDS_FIXED_AS, _T(\" Kb/s\")));\n        m_unit_combo.AddString(CCommon::LoadText(IDS_FIXED_AS, _T(\" Mb/s\")));\n    }\n    m_unit_combo.SetCurSel(static_cast<int>(m_data.speed_unit));\n}\n\nvoid CTaskBarSettingsDlg::ApplyDefaultStyle(int index)\n{\n    theApp.m_taskbar_default_style.ApplyDefaultStyle(index, m_data);\n    DrawStaticColor();\n    ((CButton*)GetDlgItem(IDC_SPECIFY_EACH_ITEM_COLOR_CHECK))->SetCheck(m_data.specify_each_item_color);\n    m_background_transparent_chk.SetCheck(IsTaskbarTransparent());\n}\n\nvoid CTaskBarSettingsDlg::ModifyDefaultStyle(int index)\n{\n    theApp.m_taskbar_default_style.ModifyDefaultStyle(index, m_data);\n}\n\nvoid CTaskBarSettingsDlg::EnableControl()\n{\n    bool exe_path_enable = (m_data.double_click_action == DoubleClickAction::SEPCIFIC_APP);\n    ShowDlgCtrl(IDC_EXE_PATH_STATIC, exe_path_enable);\n    ShowDlgCtrl(IDC_EXE_PATH_EDIT, exe_path_enable);\n    ShowDlgCtrl(IDC_BROWSE_BUTTON, exe_path_enable);\n    EnableDlgCtrl(IDC_AUTO_ADAPT_SETTINGS_BUTTON, m_data.auto_adapt_light_theme);\n    EnableDlgCtrl(IDC_SHOW_DASHED_BOX, m_data.show_status_bar || m_data.show_netspeed_figure);\n    m_status_bar_color_static.EnableWindow(m_data.show_status_bar || m_data.show_netspeed_figure);\n    EnableDlgCtrl(IDC_CM_GRAPH_BAR_RADIO, m_data.show_status_bar || m_data.show_netspeed_figure);\n    EnableDlgCtrl(IDC_CM_GRAPH_PLOT_RADIO, m_data.show_status_bar || m_data.show_netspeed_figure);\n    EnableDlgCtrl(IDC_NET_SPEED_FIGURE_MAX_VALUE_EDIT, m_data.show_netspeed_figure);\n    EnableDlgCtrl(IDC_NET_SPEED_FIGURE_MAX_VALUE_UNIT_COMBO, m_data.show_netspeed_figure);\n    //EnableDlgCtrl(IDC_TASKBAR_WND_SNAP_CHECK, theApp.m_win_version.IsWindows11OrLater() && !m_data.tbar_wnd_on_left);\n}\n\nvoid CTaskBarSettingsDlg::SetTaskabrTransparent(bool transparent)\n{\n    CTaskbarDefaultStyle::SetTaskabrTransparent(transparent, m_data);\n}\n\nbool CTaskBarSettingsDlg::IsTaskbarTransparent()\n{\n    return CTaskbarDefaultStyle::IsTaskbarTransparent(m_data);\n}\n\nvoid CTaskBarSettingsDlg::SetControlMouseWheelEnable(bool enable)\n{\n    m_unit_combo.SetMouseWheelEnable(enable);\n    m_double_click_combo.SetMouseWheelEnable(enable);\n    m_digit_number_combo.SetMouseWheelEnable(enable);\n    m_font_size_edit.SetMouseWheelEnable(enable);\n    m_memory_display_combo.SetMouseWheelEnable(enable);\n    m_item_space_edit.SetMouseWheelEnable(enable);\n    m_net_speed_figure_max_val_edit.SetMouseWheelEnable(enable);\n    m_net_speed_figure_max_val_unit_combo.SetMouseWheelEnable(enable);\n}\n\nvoid CTaskBarSettingsDlg::DoDataExchange(CDataExchange* pDX)\n{\n    DDX_Control(pDX, IDC_TEXT_COLOR_STATIC1, m_text_color_static);\n    DDX_Control(pDX, IDC_TEXT_COLOR_STATIC2, m_back_color_static);\n    DDX_Control(pDX, IDC_TEXT_COLOR_STATIC3, m_status_bar_color_static);\n    CTabDlg::DoDataExchange(pDX);\n    DDX_Control(pDX, IDC_UNIT_COMBO, m_unit_combo);\n    DDX_Control(pDX, IDC_HIDE_UNIT_CHECK, m_hide_unit_chk);\n    DDX_Control(pDX, IDC_FONT_SIZE_EDIT1, m_font_size_edit);\n    DDX_Control(pDX, IDC_DOUBLE_CLICK_COMBO, m_double_click_combo);\n    DDX_Control(pDX, IDC_DIGIT_NUMBER_COMBO, m_digit_number_combo);\n    //DDX_Control(pDX, IDC_TRANSPARENT_COLOR_STATIC, m_trans_color_static);\n    DDX_Control(pDX, IDC_BACKGROUND_TRANSPARENT_CHECK, m_background_transparent_chk);\n    DDX_Control(pDX, IDC_AUTO_ADAPT_LIGHT_THEME_CHECK, m_atuo_adapt_light_theme_chk);\n    DDX_Control(pDX, IDC_AUTO_SET_BACK_COLOR_CHECK, m_auto_set_back_color_chk);\n    DDX_Control(pDX, IDC_MEMORY_DISPLAY_COMBO, m_memory_display_combo);\n    DDX_Control(pDX, IDC_ITEM_SPACE_EDIT, m_item_space_edit);\n    DDX_Control(pDX, IDC_NET_SPEED_FIGURE_MAX_VALUE_EDIT, m_net_speed_figure_max_val_edit);\n    DDX_Control(pDX, IDC_NET_SPEED_FIGURE_MAX_VALUE_UNIT_COMBO, m_net_speed_figure_max_val_unit_combo);\n}\n\n\nBEGIN_MESSAGE_MAP(CTaskBarSettingsDlg, CTabDlg)\n    ON_BN_CLICKED(IDC_SET_FONT_BUTTON1, &CTaskBarSettingsDlg::OnBnClickedSetFontButton1)\n    ON_BN_CLICKED(IDC_TASKBAR_WND_ON_LEFT_CHECK, &CTaskBarSettingsDlg::OnBnClickedTaskbarWndOnLeftCheck)\n    ON_BN_CLICKED(IDC_SPEED_SHORT_MODE_CHECK, &CTaskBarSettingsDlg::OnBnClickedSpeedShortModeCheck)\n    ON_CBN_SELCHANGE(IDC_UNIT_COMBO, &CTaskBarSettingsDlg::OnCbnSelchangeUnitCombo)\n    ON_BN_CLICKED(IDC_HIDE_UNIT_CHECK, &CTaskBarSettingsDlg::OnBnClickedHideUnitCheck)\n    ON_BN_CLICKED(IDC_VALUE_RIGHT_ALIGN_CHECK, &CTaskBarSettingsDlg::OnBnClickedValueRightAlignCheck)\n    ON_BN_CLICKED(IDC_HIDE_PERCENTAGE_CHECK, &CTaskBarSettingsDlg::OnBnClickedHidePercentageCheck)\n    ON_MESSAGE(WM_STATIC_CLICKED, &CTaskBarSettingsDlg::OnStaticClicked)\n    ON_BN_CLICKED(IDC_SPECIFY_EACH_ITEM_COLOR_CHECK, &CTaskBarSettingsDlg::OnBnClickedSpecifyEachItemColorCheck)\n    ON_CBN_SELCHANGE(IDC_DOUBLE_CLICK_COMBO, &CTaskBarSettingsDlg::OnCbnSelchangeDoubleClickCombo)\n    ON_BN_CLICKED(IDC_HORIZONTAL_ARRANGE_CHECK, &CTaskBarSettingsDlg::OnBnClickedHorizontalArrangeCheck)\n    ON_BN_CLICKED(IDC_SHOW_STATUS_BAR_CHECK, &CTaskBarSettingsDlg::OnBnClickedShowStatusBarCheck)\n    ON_BN_CLICKED(IDC_SEPARATE_VALUE_UNIT_CHECK, &CTaskBarSettingsDlg::OnBnClickedSeparateValueUnitCheck)\n    ON_BN_CLICKED(IDC_UNIT_BYTE_RADIO, &CTaskBarSettingsDlg::OnBnClickedUnitByteRadio)\n    ON_BN_CLICKED(IDC_UNIT_BIT_RADIO, &CTaskBarSettingsDlg::OnBnClickedUnitBitRadio)\n    ON_BN_CLICKED(IDC_SHOW_TOOL_TIP_CHK, &CTaskBarSettingsDlg::OnBnClickedShowToolTipChk)\n    ON_BN_CLICKED(IDC_DEFAULT_STYLE_BUTTON, &CTaskBarSettingsDlg::OnBnClickedDefaultStyleButton)\n    ON_WM_DESTROY()\n    ON_BN_CLICKED(IDC_BROWSE_BUTTON, &CTaskBarSettingsDlg::OnBnClickedBrowseButton)\n    ON_BN_CLICKED(IDC_CM_GRAPH_BAR_RADIO, &CTaskBarSettingsDlg::OnBnClickedCMGraphBarRadio)\n    ON_BN_CLICKED(IDC_CM_GRAPH_PLOT_RADIO, &CTaskBarSettingsDlg::OnBnClickedCMGraphPLOTRadio)\n    ON_BN_CLICKED(IDC_BACKGROUND_TRANSPARENT_CHECK, &CTaskBarSettingsDlg::OnBnClickedBackgroundTransparentCheck)\n    ON_BN_CLICKED(IDC_AUTO_ADAPT_SETTINGS_BUTTON, &CTaskBarSettingsDlg::OnBnClickedAutoAdaptSettingsButton)\n    ON_BN_CLICKED(IDC_AUTO_ADAPT_LIGHT_THEME_CHECK, &CTaskBarSettingsDlg::OnBnClickedAutoAdaptLightThemeCheck)\n    ON_BN_CLICKED(IDC_AUTO_SET_BACK_COLOR_CHECK, &CTaskBarSettingsDlg::OnBnClickedAutoSetBackColorCheck)\n    ON_BN_CLICKED(IDC_DISPLAY_TEXT_SETTING_BUTTON, &CTaskBarSettingsDlg::OnBnClickedDisplayTextSettingButton)\n    ON_CBN_SELCHANGE(IDC_MEMORY_DISPLAY_COMBO, &CTaskBarSettingsDlg::OnCbnSelchangeMemoryDisplayCombo)\n    ON_BN_CLICKED(IDC_SHOW_DASHED_BOX, &CTaskBarSettingsDlg::OnBnClickedShowDashedBox)\n    ON_BN_CLICKED(IDC_SET_ORDER_BUTTON, &CTaskBarSettingsDlg::OnBnClickedSetOrderButton)\n    ON_BN_CLICKED(IDC_TASKBAR_WND_SNAP_CHECK, &CTaskBarSettingsDlg::OnBnClickedTaskbarWndSnapCheck)\n    ON_EN_CHANGE(IDC_ITEM_SPACE_EDIT, &CTaskBarSettingsDlg::OnEnChangeItemSpaceEdit)\n    ON_BN_CLICKED(IDC_SHOW_NET_SPEED_FIGURE_CHECK, &CTaskBarSettingsDlg::OnBnClickedShowNetSpeedFigureCheck)\n    ON_CBN_SELCHANGE(IDC_NET_SPEED_FIGURE_MAX_VALUE_UNIT_COMBO, &CTaskBarSettingsDlg::OnCbnSelchangeNetSpeedFigureMaxValueUnitCombo)\n    ON_EN_CHANGE(IDC_NET_SPEED_FIGURE_MAX_VALUE_EDIT, &CTaskBarSettingsDlg::OnEnChangeNetSpeedFigureMaxValueEdit)\nEND_MESSAGE_MAP()\n\n\n// CTaskBarSettingsDlg 消息处理程序\n\n\nBOOL CTaskBarSettingsDlg::OnInitDialog()\n{\n    CTabDlg::OnInitDialog();\n\n    // TODO:  在此添加额外的初始化\n\n    theApp.m_taskbar_default_style.LoadConfig();\n\n    //初始化各控件状态\n    SetDlgItemText(IDC_FONT_NAME_EDIT1, m_data.font.name);\n    //wchar_t buff[16];\n    //swprintf_s(buff, L\"%d\", m_data.font_size);\n    //SetDlgItemText(IDC_FONT_SIZE_EDIT1, buff);\n    m_font_size_edit.SetRange(5, 72);\n    m_font_size_edit.SetValue(m_data.font.size);\n\n    //SetDlgItemText(IDC_UPLOAD_EDIT1, m_data.disp_str.Get(TDI_UP).c_str());\n    //SetDlgItemText(IDC_DOWNLOAD_EDIT1, m_data.disp_str.Get(TDI_DOWN).c_str());\n    //SetDlgItemText(IDC_CPU_EDIT1, m_data.disp_str.Get(TDI_CPU).c_str());\n    //SetDlgItemText(IDC_MEMORY_EDIT1, m_data.disp_str.Get(TDI_MEMORY).c_str());\n\n    //((CButton*)GetDlgItem(IDC_SWITCH_UP_DOWN_CHECK1))->SetCheck(m_data.swap_up_down);\n    ((CButton*)GetDlgItem(IDC_TASKBAR_WND_ON_LEFT_CHECK))->SetCheck(m_data.tbar_wnd_on_left);\n    ((CButton*)GetDlgItem(IDC_SPEED_SHORT_MODE_CHECK))->SetCheck(m_data.speed_short_mode);\n    ((CButton*)GetDlgItem(IDC_VALUE_RIGHT_ALIGN_CHECK))->SetCheck(m_data.value_right_align);\n    ((CButton*)GetDlgItem(IDC_HORIZONTAL_ARRANGE_CHECK))->SetCheck(m_data.horizontal_arrange);\n    ((CButton*)GetDlgItem(IDC_SHOW_STATUS_BAR_CHECK))->SetCheck(m_data.show_status_bar);\n    ((CButton*)GetDlgItem(IDC_SEPARATE_VALUE_UNIT_CHECK))->SetCheck(m_data.separate_value_unit_with_space);\n    ((CButton*)GetDlgItem(IDC_SHOW_TOOL_TIP_CHK))->SetCheck(m_data.show_tool_tip);\n\n    EnableDlgCtrl(IDC_TASKBAR_WND_SNAP_CHECK, theApp.m_win_version.IsWindows11OrLater());\n    CheckDlgButton(IDC_TASKBAR_WND_SNAP_CHECK, m_data.tbar_wnd_snap);\n\n    m_text_color_static.SetLinkCursor();\n    m_back_color_static.SetLinkCursor();\n    //m_trans_color_static.SetLinkCursor();\n    m_status_bar_color_static.SetLinkCursor();\n    DrawStaticColor();\n\n#ifdef COMPILE_FOR_WINXP\n    m_background_transparent_chk.EnableWindow(FALSE);\n#endif // COMPILE_FOR_WINXP\n\n    if (theApp.m_win_version.IsWindows7())\n        m_background_transparent_chk.EnableWindow(FALSE);\n\n    m_toolTip.Create(this);\n    m_toolTip.SetMaxTipWidth(theApp.DPI(300));\n    m_toolTip.AddTool(GetDlgItem(IDC_SPEED_SHORT_MODE_CHECK), CCommon::LoadText(IDS_SPEED_SHORT_MODE_TIP));\n    m_toolTip.AddTool(&m_atuo_adapt_light_theme_chk, CCommon::LoadText(IDS_AUTO_ADAPT_TIP_INFO));\n    m_toolTip.AddTool(GetDlgItem(IDC_SHOW_STATUS_BAR_CHECK), CCommon::LoadText(IDS_SHOW_RESOURCE_USAGE_GRAPH_TIP));\n    m_toolTip.AddTool(GetDlgItem(IDC_SHOW_NET_SPEED_FIGURE_CHECK), CCommon::LoadText(IDS_SHOW_NET_SPEED_GRAPH_TIP));\n\n\n    if (m_data.unit_byte)\n        ((CButton*)GetDlgItem(IDC_UNIT_BYTE_RADIO))->SetCheck(TRUE);\n    else\n        ((CButton*)GetDlgItem(IDC_UNIT_BIT_RADIO))->SetCheck(TRUE);\n\n    IniUnitCombo();\n\n    m_hide_unit_chk.SetCheck(m_data.hide_unit);\n    if (m_data.speed_unit == SpeedUnit::AUTO)\n    {\n        m_hide_unit_chk.SetCheck(FALSE);\n        m_data.hide_unit = false;\n        m_hide_unit_chk.EnableWindow(FALSE);\n    }\n    ((CButton*)GetDlgItem(IDC_HIDE_PERCENTAGE_CHECK))->SetCheck(m_data.hide_percent);\n    ((CButton*)GetDlgItem(IDC_SPECIFY_EACH_ITEM_COLOR_CHECK))->SetCheck(m_data.specify_each_item_color);\n    m_background_transparent_chk.SetCheck(IsTaskbarTransparent());\n    m_atuo_adapt_light_theme_chk.SetCheck(m_data.auto_adapt_light_theme);\n    m_auto_set_back_color_chk.SetCheck(m_data.auto_set_background_color);\n    m_auto_set_back_color_chk.EnableWindow(theApp.m_win_version.IsWindows8OrLater());\n\n    if (theApp.m_win_version.GetMajorVersion() < 10)\n    {\n        m_data.auto_adapt_light_theme = false;\n        m_atuo_adapt_light_theme_chk.EnableWindow(FALSE);\n    }\n\n    m_double_click_combo.AddString(CCommon::LoadText(IDS_OPEN_CONNECTION_DETIAL));\n    m_double_click_combo.AddString(CCommon::LoadText(IDS_OPEN_HISTORICAL_TRAFFIC));\n    m_double_click_combo.AddString(CCommon::LoadText(IDS_SHOW_HIDE_CPU_MEMORY));\n    m_double_click_combo.AddString(CCommon::LoadText(IDS_OPEN_OPTION_SETTINGS));\n    m_double_click_combo.AddString(CCommon::LoadText(IDS_OPEN_TASK_MANAGER));\n    m_double_click_combo.AddString(CCommon::LoadText(IDS_SPECIFIC_APP));\n    m_double_click_combo.AddString(CCommon::LoadText(IDS_NONE));\n    m_double_click_combo.SetCurSel(static_cast<int>(m_data.double_click_action));\n\n    m_digit_number_combo.AddString(_T(\"3\"));\n    m_digit_number_combo.AddString(_T(\"4\"));\n    m_digit_number_combo.AddString(_T(\"5\"));\n    m_digit_number_combo.AddString(_T(\"6\"));\n    m_digit_number_combo.AddString(_T(\"7\"));\n    m_digit_number_combo.SetCurSel(m_data.digits_number - 3);\n\n    SetDlgItemText(IDC_EXE_PATH_EDIT, m_data.double_click_exe.c_str());\n    EnableControl();\n\n    //m_default_style_menu.LoadMenu(IDR_TASKBAR_STYLE_MENU);\n\n    if (m_data.cm_graph_type)\n        CheckDlgButton(IDC_CM_GRAPH_PLOT_RADIO, TRUE);\n    else\n        CheckDlgButton(IDC_CM_GRAPH_BAR_RADIO, TRUE);\n    CheckDlgButton(IDC_SHOW_DASHED_BOX, m_data.show_graph_dashed_box);\n    m_item_space_edit.SetRange(0, 32);\n    m_item_space_edit.SetValue(m_data.item_space);\n\n    //初始化内存显示方式下拉列表\n    m_memory_display_combo.AddString(CCommon::LoadText(IDS_USAGE_PERCENTAGE));\n    m_memory_display_combo.AddString(CCommon::LoadText(IDS_MEMORY_USED));\n    m_memory_display_combo.AddString(CCommon::LoadText(IDS_MEMORY_AVAILABLE));\n    m_memory_display_combo.SetCurSel(static_cast<int>(m_data.memory_display));\n\n    CheckDlgButton(IDC_SHOW_NET_SPEED_FIGURE_CHECK, m_data.show_netspeed_figure);\n    m_net_speed_figure_max_val_edit.SetRange(1, 1024);\n    m_net_speed_figure_max_val_edit.SetValue(m_data.netspeed_figure_max_value);\n    m_net_speed_figure_max_val_unit_combo.AddString(_T(\"KB\"));\n    m_net_speed_figure_max_val_unit_combo.AddString(_T(\"MB\"));\n    m_net_speed_figure_max_val_unit_combo.SetCurSel(m_data.netspeed_figure_max_value_unit);\n\n    ((CButton*)GetDlgItem(IDC_SET_ORDER_BUTTON))->SetIcon(theApp.GetMenuIcon(IDI_ITEM));\n\n    //初始化“预设方案”菜单\n    m_default_style_menu.CreatePopupMenu();\n    m_modify_default_style_menu.CreatePopupMenu();\n    for (int i{ 0 }; i < TASKBAR_DEFAULT_STYLE_NUM; i++)\n    {\n        CString item_name;\n        item_name.Format(_T(\"%s %d\"), CCommon::LoadText(IDS_PRESET).GetString(), i + 1);\n        m_default_style_menu.AppendMenu(MF_STRING | MF_ENABLED, ID_DEFAULT_STYLE1 + i, item_name);\n        m_modify_default_style_menu.AppendMenu(MF_STRING | MF_ENABLED, ID_MODIFY_DEFAULT_STYLE1 + i, item_name);\n    }\n    m_default_style_menu.AppendMenu(MF_SEPARATOR);\n    m_default_style_menu.AppendMenu(MF_POPUP | MF_STRING, (UINT)m_modify_default_style_menu.m_hMenu, CCommon::LoadText(IDS_MODIFY_PRESET));\n\n    return TRUE;  // return TRUE unless you set the focus to a control\n                  // 异常: OCX 属性页应返回 FALSE\n}\n\n\nvoid CTaskBarSettingsDlg::OnBnClickedSetFontButton1()\n{\n    // TODO: 在此添加控件通知处理程序代码\n    LOGFONT lf{};\n    lf.lfHeight = FontSizeToLfHeight(m_data.font.size);\n    lf.lfWeight = (m_data.font.bold ? FW_BOLD : FW_NORMAL);\n    lf.lfItalic = m_data.font.italic;\n    lf.lfUnderline = m_data.font.underline;\n    lf.lfStrikeOut = m_data.font.strike_out;\n    lf.lfPitchAndFamily = DEFAULT_PITCH | FF_SWISS;\n    //wcsncpy_s(lf.lfFaceName, m_data.font.name.GetString(), 32);\n    CCommon::WStringCopy(lf.lfFaceName, 32, m_data.font.name.GetString());\n    CCommon::NormalizeFont(lf);\n    CFontDialog fontDlg(&lf);   //构造字体对话框，初始选择字体为之前字体\n    if (IDOK == fontDlg.DoModal())     // 显示字体对话框\n    {\n        //获取字体信息\n        m_data.font.name = fontDlg.GetFaceName();\n        m_data.font.size = fontDlg.GetSize() / 10;\n        m_data.font.bold = (fontDlg.IsBold() != FALSE);\n        m_data.font.italic = (fontDlg.IsItalic() != FALSE);\n        m_data.font.underline = (fontDlg.IsUnderline() != FALSE);\n        m_data.font.strike_out = (fontDlg.IsStrikeOut() != FALSE);\n        //将字体信息显示出来\n        SetDlgItemText(IDC_FONT_NAME_EDIT1, m_data.font.name);\n        wchar_t buff[16];\n        swprintf_s(buff, L\"%d\", m_data.font.size);\n        SetDlgItemText(IDC_FONT_SIZE_EDIT1, buff);\n    }\n}\n\n\nvoid CTaskBarSettingsDlg::OnBnClickedTaskbarWndOnLeftCheck()\n{\n    // TODO: 在此添加控件通知处理程序代码\n    m_data.tbar_wnd_on_left = (((CButton*)GetDlgItem(IDC_TASKBAR_WND_ON_LEFT_CHECK))->GetCheck() != 0);\n    EnableControl();\n}\n\n\nvoid CTaskBarSettingsDlg::OnBnClickedSpeedShortModeCheck()\n{\n    // TODO: 在此添加控件通知处理程序代码\n    m_data.speed_short_mode = (((CButton*)GetDlgItem(IDC_SPEED_SHORT_MODE_CHECK))->GetCheck() != 0);\n}\n\n\nBOOL CTaskBarSettingsDlg::PreTranslateMessage(MSG* pMsg)\n{\n    // TODO: 在此添加专用代码和/或调用基类\n    if (pMsg->message == WM_MOUSEMOVE)\n        m_toolTip.RelayEvent(pMsg);\n\n    return CTabDlg::PreTranslateMessage(pMsg);\n}\n\n\nvoid CTaskBarSettingsDlg::OnCbnSelchangeUnitCombo()\n{\n    // TODO: 在此添加控件通知处理程序代码\n    m_data.speed_unit = static_cast<SpeedUnit>(m_unit_combo.GetCurSel());\n    if (m_data.speed_unit == SpeedUnit::AUTO)\n    {\n        m_hide_unit_chk.SetCheck(FALSE);\n        m_data.hide_unit = false;\n        m_hide_unit_chk.EnableWindow(FALSE);\n    }\n    else\n    {\n        m_hide_unit_chk.EnableWindow(TRUE);\n    }\n}\n\n\nvoid CTaskBarSettingsDlg::OnBnClickedHideUnitCheck()\n{\n    // TODO: 在此添加控件通知处理程序代码\n    m_data.hide_unit = (m_hide_unit_chk.GetCheck() != 0);\n}\n\n\nvoid CTaskBarSettingsDlg::OnOK()\n{\n    // TODO: 在此添加专用代码和/或调用基类\n    //获取字体设置\n    int font_size;\n    font_size = m_font_size_edit.GetValue();\n    if (font_size > MAX_FONT_SIZE || font_size < MIN_FONT_SIZE)\n    {\n        CString info;\n        info.Format(CCommon::LoadText(IDS_FONT_SIZE_WARNING), MIN_FONT_SIZE, MAX_FONT_SIZE);\n        MessageBox(info, NULL, MB_OK | MB_ICONWARNING);\n    }\n    else\n    {\n        m_data.font.size = font_size;\n    }\n    GetDlgItemText(IDC_FONT_NAME_EDIT1, m_data.font.name);\n\n    //获取数据位数的设置\n    m_data.digits_number = m_digit_number_combo.GetCurSel() + 3;\n\n    bool is_taskbar_transparent_checked = (m_background_transparent_chk.GetCheck() != 0);\n    SetTaskabrTransparent(is_taskbar_transparent_checked);\n\n    SaveColorSettingToDefaultStyle();\n\n    CTabDlg::OnOK();\n}\n\n\nvoid CTaskBarSettingsDlg::SaveColorSettingToDefaultStyle()\n{\n    //如果开启了自动适应Windows10深色/浅色模式功能时，自动将当前配置保存到对应预设\n    if (theApp.m_taskbar_data.auto_save_taskbar_color_settings_to_preset && m_data.auto_adapt_light_theme && IsStyleModified())\n    {\n        int default_style_saved{ CWindowsSettingHelper::IsWindows10LightTheme() ? m_data.light_default_style : m_data.dark_default_style };\n        ModifyDefaultStyle(default_style_saved);\n    }\n}\n\nvoid CTaskBarSettingsDlg::OnBnClickedValueRightAlignCheck()\n{\n    // TODO: 在此添加控件通知处理程序代码\n    m_data.value_right_align = (((CButton*)GetDlgItem(IDC_VALUE_RIGHT_ALIGN_CHECK))->GetCheck() != 0);\n}\n\n\nvoid CTaskBarSettingsDlg::OnBnClickedHidePercentageCheck()\n{\n    // TODO: 在此添加控件通知处理程序代码\n    m_data.hide_percent = (((CButton*)GetDlgItem(IDC_HIDE_PERCENTAGE_CHECK))->GetCheck() != 0);\n}\n\n\nafx_msg LRESULT CTaskBarSettingsDlg::OnStaticClicked(WPARAM wParam, LPARAM lParam)\n{\n    switch (::GetDlgCtrlID(((CWnd*)wParam)->m_hWnd))\n    {\n    case IDC_TEXT_COLOR_STATIC1:        //点击“文本颜色”时\n    {\n        //设置文本颜色\n        if (m_data.specify_each_item_color)\n        {\n            CTaskbarColorDlg colorDlg(m_data.text_colors);\n            if (colorDlg.DoModal() == IDOK)\n            {\n                m_data.text_colors = colorDlg.GetColors();\n                DrawStaticColor();\n                m_style_modified = true;\n            }\n        }\n        else if (!m_data.text_colors.empty())\n        {\n            CMFCColorDialogEx colorDlg(m_data.text_colors.begin()->second.label, 0, this);\n            if (colorDlg.DoModal() == IDOK)\n            {\n                m_data.text_colors.begin()->second.label = colorDlg.GetColor();\n                if (m_data.back_color == m_data.text_colors.begin()->second.label)\n                    MessageBox(CCommon::LoadText(IDS_SAME_TEXT_BACK_COLOR_WARNING), NULL, MB_ICONWARNING);\n                DrawStaticColor();\n                m_style_modified = true;\n            }\n        }\n        break;\n    }\n    case IDC_TEXT_COLOR_STATIC2:        //点击“背景颜色”时\n    {\n        //设置背景颜色\n        CMFCColorDialogEx colorDlg(m_data.back_color, 0, this);\n        if (colorDlg.DoModal() == IDOK)\n        {\n            bool background_transparent = IsTaskbarTransparent();\n            m_data.back_color = colorDlg.GetColor();\n            if (m_data.back_color == m_data.text_colors.begin()->second.label)\n                MessageBox(CCommon::LoadText(IDS_SAME_BACK_TEXT_COLOR_WARNING), NULL, MB_ICONWARNING);\n            if (background_transparent)\n            {\n                CCommon::TransparentColorConvert(m_data.back_color);\n                //如果当前设置了背景透明，则更改了背景色后同时将透明色设置成和背景色一样的颜色，以保持背景透明\n                m_data.transparent_color = m_data.back_color;\n            }\n            DrawStaticColor();\n            m_style_modified = true;\n        }\n        break;\n    }\n    //case IDC_TRANSPARENT_COLOR_STATIC:        //点击“透明色”时\n    //{\n    //  CMFCColorDialogEx colorDlg(m_data.transparent_color, 0, this);\n    //  if (colorDlg.DoModal() == IDOK)\n    //  {\n    //      m_data.transparent_color = colorDlg.GetColor();\n    //      DrawStaticColor();\n    //  }\n    //  break;\n    //}\n    case IDC_TEXT_COLOR_STATIC3:        //点击“状态条颜色”时\n    {\n        CMFCColorDialogEx colorDlg(m_data.status_bar_color, 0, this);\n        if (colorDlg.DoModal() == IDOK)\n        {\n            m_data.status_bar_color = colorDlg.GetColor();\n            DrawStaticColor();\n            m_style_modified = true;\n        }\n        break;\n    }\n    default:\n        break;\n    }\n    return 0;\n}\n\n\nvoid CTaskBarSettingsDlg::OnBnClickedSpecifyEachItemColorCheck()\n{\n    // TODO: 在此添加控件通知处理程序代码\n    m_data.specify_each_item_color = (((CButton*)GetDlgItem(IDC_SPECIFY_EACH_ITEM_COLOR_CHECK))->GetCheck() != 0);\n    DrawStaticColor();\n    m_style_modified = true;\n}\n\n\nvoid CTaskBarSettingsDlg::OnCbnSelchangeDoubleClickCombo()\n{\n    // TODO: 在此添加控件通知处理程序代码\n    m_data.double_click_action = static_cast<DoubleClickAction>(m_double_click_combo.GetCurSel());\n    EnableControl();\n}\n\n\nvoid CTaskBarSettingsDlg::OnBnClickedHorizontalArrangeCheck()\n{\n    // TODO: 在此添加控件通知处理程序代码\n    m_data.horizontal_arrange = (((CButton*)GetDlgItem(IDC_HORIZONTAL_ARRANGE_CHECK))->GetCheck() != 0);\n}\n\n\nvoid CTaskBarSettingsDlg::OnBnClickedShowStatusBarCheck()\n{\n    // TODO: 在此添加控件通知处理程序代码\n    m_data.show_status_bar = (((CButton*)GetDlgItem(IDC_SHOW_STATUS_BAR_CHECK))->GetCheck() != 0);\n    EnableControl();\n}\n\n\nvoid CTaskBarSettingsDlg::OnBnClickedSeparateValueUnitCheck()\n{\n    // TODO: 在此添加控件通知处理程序代码\n    m_data.separate_value_unit_with_space = (((CButton*)GetDlgItem(IDC_SEPARATE_VALUE_UNIT_CHECK))->GetCheck() != 0);\n}\n\n\nvoid CTaskBarSettingsDlg::OnBnClickedUnitByteRadio()\n{\n    // TODO: 在此添加控件通知处理程序代码\n    m_data.unit_byte = true;\n    IniUnitCombo();\n}\n\n\nvoid CTaskBarSettingsDlg::OnBnClickedUnitBitRadio()\n{\n    // TODO: 在此添加控件通知处理程序代码\n    m_data.unit_byte = false;\n    IniUnitCombo();\n}\n\n\nvoid CTaskBarSettingsDlg::OnBnClickedShowToolTipChk()\n{\n    // TODO: 在此添加控件通知处理程序代码\n    m_data.show_tool_tip = (((CButton*)GetDlgItem(IDC_SHOW_TOOL_TIP_CHK))->GetCheck() != 0);\n}\n\n\nvoid CTaskBarSettingsDlg::OnBnClickedDefaultStyleButton()\n{\n    // TODO: 在此添加控件通知处理程序代码\n    CWnd* pBtn = GetDlgItem(IDC_DEFAULT_STYLE_BUTTON);\n    CPoint point;\n    if (pBtn != nullptr)\n    {\n        CRect rect;\n        pBtn->GetWindowRect(rect);\n        point.x = rect.left;\n        point.y = rect.bottom;\n        m_default_style_menu.TrackPopupMenu(TPM_LEFTALIGN | TPM_RIGHTBUTTON, point.x, point.y, this);\n    }\n}\n\n\nvoid CTaskBarSettingsDlg::OnDestroy()\n{\n    CTabDlg::OnDestroy();\n\n    // TODO: 在此处添加消息处理程序代码\n    theApp.m_taskbar_default_style.SaveConfig();\n}\n\n\nvoid CTaskBarSettingsDlg::OnBnClickedBrowseButton()\n{\n    // TODO: 在此添加控件通知处理程序代码\n    CString szFilter = CCommon::LoadText(IDS_EXE_FILTER);\n    CFileDialog fileDlg(TRUE, NULL, NULL, 0, szFilter, this);\n    if (IDOK == fileDlg.DoModal())\n    {\n        m_data.double_click_exe = fileDlg.GetPathName();\n        SetDlgItemText(IDC_EXE_PATH_EDIT, m_data.double_click_exe.c_str());\n    }\n}\n\nvoid CTaskBarSettingsDlg::OnBnClickedCMGraphPLOTRadio()\n{\n    m_data.cm_graph_type = true;\n}\n\nvoid CTaskBarSettingsDlg::OnBnClickedCMGraphBarRadio()\n{\n    m_data.cm_graph_type = false;\n}\n\n\nvoid CTaskBarSettingsDlg::OnBnClickedBackgroundTransparentCheck()\n{\n    // TODO: 在此添加控件通知处理程序代码\n    bool checked = (m_background_transparent_chk.GetCheck() != 0);\n    SetTaskabrTransparent(checked);\n    m_style_modified = true;\n}\n\n\nvoid CTaskBarSettingsDlg::OnBnClickedAutoAdaptSettingsButton()\n{\n    // TODO: 在此添加控件通知处理程序代码\n    CAutoAdaptSettingsDlg dlg(m_data);\n    dlg.DoModal();\n}\n\n\nvoid CTaskBarSettingsDlg::OnBnClickedAutoAdaptLightThemeCheck()\n{\n    // TODO: 在此添加控件通知处理程序代码\n    m_data.auto_adapt_light_theme = (m_atuo_adapt_light_theme_chk.GetCheck() != 0);\n    EnableControl();\n}\n\n\nvoid CTaskBarSettingsDlg::OnBnClickedAutoSetBackColorCheck()\n{\n    // TODO: 在此添加控件通知处理程序代码\n    m_data.auto_set_background_color = (m_auto_set_back_color_chk.GetCheck() != 0);\n}\n\n\nvoid CTaskBarSettingsDlg::OnBnClickedDisplayTextSettingButton()\n{\n    // TODO: 在此添加控件通知处理程序代码\n    CDisplayTextSettingDlg dlg(m_data.disp_str);\n    dlg.DoModal();\n}\n\n\nvoid CTaskBarSettingsDlg::OnCbnSelchangeMemoryDisplayCombo()\n{\n    // TODO: 在此添加控件通知处理程序代码\n    m_data.memory_display = static_cast<MemoryDisplay>(m_memory_display_combo.GetCurSel());\n}\n\n\nvoid CTaskBarSettingsDlg::OnBnClickedShowDashedBox()\n{\n    // TODO: 在此添加控件通知处理程序代码\n    m_data.show_graph_dashed_box = (IsDlgButtonChecked(IDC_SHOW_DASHED_BOX) != 0);\n}\n\n\nvoid CTaskBarSettingsDlg::OnBnClickedSetOrderButton()\n{\n    // TODO: 在此添加控件通知处理程序代码\n    CSetItemOrderDlg dlg;\n    dlg.SetItemOrder(m_data.item_order.GetItemOrderConst());\n    dlg.SetDisplayItem(m_data.m_tbar_display_item);\n    dlg.SetPluginDisplayItem(m_data.plugin_display_item);\n    if (dlg.DoModal() == IDOK)\n    {\n        m_data.item_order.SetOrder(dlg.GetItemOrder());\n        m_data.m_tbar_display_item = dlg.GetDisplayItem();\n        m_data.plugin_display_item = dlg.GetPluginDisplayItem();\n    }\n}\n\n\nvoid CTaskBarSettingsDlg::OnBnClickedTaskbarWndSnapCheck()\n{\n    // TODO: 在此添加控件通知处理程序代码\n    m_data.tbar_wnd_snap = (IsDlgButtonChecked(IDC_TASKBAR_WND_SNAP_CHECK) != 0);\n}\n\n\nvoid CTaskBarSettingsDlg::OnEnChangeItemSpaceEdit()\n{\n    // TODO:  如果该控件是 RICHEDIT 控件，它将不\n    // 发送此通知，除非重写 CTabDlg::OnInitDialog()\n    // 函数并调用 CRichEditCtrl().SetEventMask()，\n    // 同时将 ENM_CHANGE 标志“或”运算到掩码中。\n\n    // TODO:  在此添加控件通知处理程序代码\n    m_data.item_space = m_item_space_edit.GetValue();\n    m_data.ValidItemSpace();\n}\n\n\nBOOL CTaskBarSettingsDlg::OnCommand(WPARAM wParam, LPARAM lParam)\n{\n    // TODO: 在此添加专用代码和/或调用基类\n    UINT cmd = LOWORD(wParam);\n\n    if (cmd >= ID_DEFAULT_STYLE1 && cmd < ID_DEFAULT_STYLE_MAX)\n    {\n        int default_style = cmd - ID_DEFAULT_STYLE1;\n        ApplyDefaultStyle(default_style);\n    }\n    if (cmd >= ID_MODIFY_DEFAULT_STYLE1 && cmd < ID_MODIFY_DEFAULT_STYLE_MAX)\n    {\n        int default_style = cmd - ID_MODIFY_DEFAULT_STYLE1;\n        if (MessageBox(CCommon::LoadTextFormat(IDS_SAVE_DEFAULT_STYLE_INQUIRY, { default_style + 1 }), NULL, MB_ICONQUESTION | MB_YESNO) == IDYES)\n        {\n            ModifyDefaultStyle(default_style);\n        }\n    }\n\n    return CTabDlg::OnCommand(wParam, lParam);\n}\n\n\nvoid CTaskBarSettingsDlg::OnBnClickedShowNetSpeedFigureCheck()\n{\n    m_data.show_netspeed_figure = (IsDlgButtonChecked(IDC_SHOW_NET_SPEED_FIGURE_CHECK) != 0);\n    EnableControl();\n}\n\n\nvoid CTaskBarSettingsDlg::OnCbnSelchangeNetSpeedFigureMaxValueUnitCombo()\n{\n    m_data.netspeed_figure_max_value_unit = m_net_speed_figure_max_val_unit_combo.GetCurSel();\n}\n\n\nvoid CTaskBarSettingsDlg::OnEnChangeNetSpeedFigureMaxValueEdit()\n{\n    m_data.netspeed_figure_max_value = m_net_speed_figure_max_val_edit.GetValue();\n}\n"
  },
  {
    "path": "TrafficMonitor/TaskBarSettingsDlg.h",
    "content": "﻿#pragma once\r\n#include \"ColorStatic.h\"\r\n#include \"afxwin.h\"\r\n#include \"SpinEdit.h\"\r\n#include \"TabDlg.h\"\r\n#include \"TaskbarColorDlg.h\"\r\n#include \"ComboBox2.h\"\r\n\r\n// CTaskBarSettingsDlg 对话框\r\n\r\nclass CTaskBarSettingsDlg : public CTabDlg\r\n{\r\n    DECLARE_DYNAMIC(CTaskBarSettingsDlg)\r\n\r\npublic:\r\n    CTaskBarSettingsDlg(CWnd* pParent = NULL);   // 标准构造函数\r\n    virtual ~CTaskBarSettingsDlg();\r\n\r\n    bool IsStyleModified();\r\n\r\n    //如果开启了自动适应Windows10深色/浅色模式功能时，将当前配置保存到对应预设\r\n    void SaveColorSettingToDefaultStyle();\r\n\r\n    //选项设置数据\r\n    TaskBarSettingData m_data;\r\n\r\n\tCWinVersionHelper m_win_version;        //当前Windows的版本\r\n\r\n    // 对话框数据\r\n#ifdef AFX_DESIGN_TIME\r\n    enum { IDD = IDD_TASKBAR_SETTINGS_DIALOG };\r\n#endif\r\n\r\nprotected:\r\n    //控件变量\r\n    CColorStatic m_text_color_static;\r\n    CColorStatic m_back_color_static;\r\n    CColorStatic m_trans_color_static;\r\n    CColorStatic m_status_bar_color_static;\r\n    CToolTipCtrl m_toolTip;\r\n    CComboBox2 m_unit_combo;\r\n    CButton m_hide_unit_chk;\r\n    CSpinEdit m_font_size_edit;\r\n    CComboBox2 m_double_click_combo;\r\n    CComboBox2 m_digit_number_combo;\r\n    CMenu m_default_style_menu;\r\n    CMenu m_modify_default_style_menu;\r\n    CButton m_background_transparent_chk;\r\n    CButton m_atuo_adapt_light_theme_chk;\r\n    CButton m_auto_set_back_color_chk;\r\n    CComboBox2 m_memory_display_combo;\r\n    CSpinEdit m_item_space_edit;\r\n    CSpinEdit m_net_speed_figure_max_val_edit;\r\n    CComboBox2 m_net_speed_figure_max_val_unit_combo;\r\n\r\n    bool m_style_modified{};\r\n\r\nprotected:\r\n    void DrawStaticColor();\r\n    void IniUnitCombo();\r\n\r\n    void ApplyDefaultStyle(int index);      //应用一个预设方案\r\n\r\npublic:\r\n    void ModifyDefaultStyle(int index);     //将当前颜色设置保存到一个预设方案\r\n\r\nprotected:\r\n    void EnableControl();\r\n\r\n    void SetTaskabrTransparent(bool transparent);\r\n    bool IsTaskbarTransparent();\r\n    virtual void SetControlMouseWheelEnable(bool enable) override;\r\n\r\n    virtual void DoDataExchange(CDataExchange* pDX);    // DDX/DDV 支持\r\n\r\n    DECLARE_MESSAGE_MAP()\r\npublic:\r\n    virtual BOOL OnInitDialog();\r\n    afx_msg void OnBnClickedSetFontButton1();\r\n    afx_msg void OnBnClickedTaskbarWndOnLeftCheck();\r\n    afx_msg void OnBnClickedSpeedShortModeCheck();\r\n    virtual BOOL PreTranslateMessage(MSG* pMsg);\r\n    afx_msg void OnCbnSelchangeUnitCombo();\r\n    afx_msg void OnBnClickedHideUnitCheck();\r\n    virtual void OnOK();\r\n    afx_msg void OnBnClickedValueRightAlignCheck();\r\n    afx_msg void OnBnClickedHidePercentageCheck();\r\nprotected:\r\n    afx_msg LRESULT OnStaticClicked(WPARAM wParam, LPARAM lParam);\r\npublic:\r\n    afx_msg void OnBnClickedSpecifyEachItemColorCheck();\r\n    afx_msg void OnCbnSelchangeDoubleClickCombo();\r\n    afx_msg void OnBnClickedHorizontalArrangeCheck();\r\n    afx_msg void OnBnClickedShowStatusBarCheck();\r\n    afx_msg void OnBnClickedSeparateValueUnitCheck();\r\n    afx_msg void OnBnClickedUnitByteRadio();\r\n    afx_msg void OnBnClickedUnitBitRadio();\r\n    afx_msg void OnBnClickedShowToolTipChk();\r\n    afx_msg void OnBnClickedDefaultStyleButton();\r\n    afx_msg void OnDestroy();\r\n    afx_msg void OnBnClickedBrowseButton();\r\n    afx_msg void OnBnClickedCMGraphBarRadio();\r\n    afx_msg void OnBnClickedCMGraphPLOTRadio();\r\n    afx_msg void OnBnClickedBackgroundTransparentCheck();\r\n    afx_msg void OnBnClickedAutoAdaptSettingsButton();\r\n    afx_msg void OnBnClickedAutoAdaptLightThemeCheck();\r\n    afx_msg void OnBnClickedAutoSetBackColorCheck();\r\n    afx_msg void OnBnClickedDisplayTextSettingButton();\r\n    afx_msg void OnCbnSelchangeMemoryDisplayCombo();\r\n    afx_msg void OnBnClickedShowDashedBox();\r\n    afx_msg void OnBnClickedSetOrderButton();\r\n    afx_msg void OnBnClickedTaskbarWndSnapCheck();\r\n    afx_msg void OnEnChangeItemSpaceEdit();\r\n    virtual BOOL OnCommand(WPARAM wParam, LPARAM lParam);\r\n    afx_msg void OnBnClickedShowNetSpeedFigureCheck();\r\n    afx_msg void OnCbnSelchangeNetSpeedFigureMaxValueUnitCombo();\r\n    afx_msg void OnEnChangeNetSpeedFigureMaxValueEdit();\r\n};\r\n"
  },
  {
    "path": "TrafficMonitor/TaskbarColorDlg.cpp",
    "content": "﻿// TaskbarColorDlg.cpp : 实现文件\n//\n\n#include \"stdafx.h\"\n#include \"TrafficMonitor.h\"\n#include \"TaskbarColorDlg.h\"\n#include \"afxdialogex.h\"\n#include \"CMFCColorDialogEx.h\"\n\n\n// CTaskbarColorDlg 对话框\n\nIMPLEMENT_DYNAMIC(CTaskbarColorDlg, CBaseDialog)\n\nCTaskbarColorDlg::CTaskbarColorDlg(const std::map<CommonDisplayItem, TaskbarItemColor>& colors, CWnd* pParent /*=NULL*/)\n\t: CBaseDialog(IDD_TASKBAR_COLOR_DIALOG, pParent), m_colors(colors)\n{\n}\n\nCTaskbarColorDlg::~CTaskbarColorDlg()\n{\n}\n\nCString CTaskbarColorDlg::GetDialogName() const\n{\n    return _T(\"TaskbarColorDlg\");\n}\n\nvoid CTaskbarColorDlg::DoDataExchange(CDataExchange* pDX)\n{\n    CBaseDialog::DoDataExchange(pDX);\n    DDX_Control(pDX, IDC_LIST1, m_list_ctrl);\n}\n\n\nBEGIN_MESSAGE_MAP(CTaskbarColorDlg, CBaseDialog)\n    ON_NOTIFY(NM_DBLCLK, IDC_LIST1, &CTaskbarColorDlg::OnNMDblclkList1)\nEND_MESSAGE_MAP()\n\n\n// CTaskbarColorDlg 消息处理程序\n\n\nBOOL CTaskbarColorDlg::OnInitDialog()\n{\n\tCBaseDialog::OnInitDialog();\n\n\t// TODO:  在此添加额外的初始化\n\n    SetIcon(theApp.GetMenuIcon(IDI_TASKBAR_WINDOW), FALSE);\t\t// 设置小图标\n\n    //初始化列表控件\n    CRect rect;\n    m_list_ctrl.GetClientRect(rect);\n    m_list_ctrl.SetExtendedStyle(LVS_EX_FULLROWSELECT | LVS_EX_LABELTIP);\n    int width0 = rect.Width() /2;\n    int width1 = (rect.Width() - width0 - theApp.DPI(20) - 1) / 2;\n    int width2 = rect.Width() - width0 - width1 - theApp.DPI(20) - 1;\n    m_list_ctrl.InsertColumn(0, CCommon::LoadText(IDS_ITEM), LVCFMT_LEFT, width0);\n    m_list_ctrl.InsertColumn(1, CCommon::LoadText(IDS_COLOR_LABEL), LVCFMT_LEFT, width1);\n    m_list_ctrl.InsertColumn(2, CCommon::LoadText(IDS_COLOR_VALUE), LVCFMT_LEFT, width2);\n    m_list_ctrl.SetDrawItemRangMargin(theApp.DPI(2));\n\n    //向列表中插入行\n    for (auto iter = theApp.m_plugins.AllDisplayItemsWithPlugins().begin(); iter != theApp.m_plugins.AllDisplayItemsWithPlugins().end(); ++iter)\n    {\n        CString item_name = iter->GetItemName();\n        if (!item_name.IsEmpty())\n        {\n            int index = m_list_ctrl.GetItemCount();\n            m_list_ctrl.InsertItem(index, item_name);\n            m_list_ctrl.SetItemColor(index, 1, m_colors[*iter].label);\n            m_list_ctrl.SetItemColor(index, 2, m_colors[*iter].value);\n            m_list_ctrl.SetItemData(index, (DWORD_PTR)&(*iter));\n        }\n    }\n\n\treturn TRUE;  // return TRUE unless you set the focus to a control\n\t\t\t\t  // 异常: OCX 属性页应返回 FALSE\n}\n\n\n\nvoid CTaskbarColorDlg::OnNMDblclkList1(NMHDR *pNMHDR, LRESULT *pResult)\n{\n    LPNMITEMACTIVATE pNMItemActivate = reinterpret_cast<LPNMITEMACTIVATE>(pNMHDR);\n    // TODO: 在此添加控件通知处理程序代码\n    int index = pNMItemActivate->iItem;\n    int col = pNMItemActivate->iSubItem;\n    if (col == 1 || col == 2)\n    {\n        COLORREF color = m_list_ctrl.GetItemColor(index, col);\n        CMFCColorDialogEx colorDlg(color, 0, this);\n        if (colorDlg.DoModal() == IDOK)\n        {\n            color = colorDlg.GetColor();\n            m_list_ctrl.SetItemColor(index, col, color);\n            CommonDisplayItem* item = (CommonDisplayItem*)(m_list_ctrl.GetItemData(index));\n            if (col == 1)\n                m_colors[*item].label = color;\n            else\n                m_colors[*item].value = color;\n        }\n    }\n\n    *pResult = 0;\n}\n"
  },
  {
    "path": "TrafficMonitor/TaskbarColorDlg.h",
    "content": "﻿#pragma once\n#include \"ColorStatic.h\"\n#include \"afxwin.h\"\n#include \"ColorSettingListCtrl.h\"\n#include \"BaseDialog.h\"\n\n// CTaskbarColorDlg 对话框\n\nclass CTaskbarColorDlg : public CBaseDialog\n{\n\tDECLARE_DYNAMIC(CTaskbarColorDlg)\n\npublic:\n\tCTaskbarColorDlg(const std::map<CommonDisplayItem, TaskbarItemColor>& colors, CWnd* pParent = NULL);   // 标准构造函数\n\tvirtual ~CTaskbarColorDlg();\n\n    const std::map<CommonDisplayItem, TaskbarItemColor>& GetColors() const { return m_colors; }\n\n\t// 对话框数据\n#ifdef AFX_DESIGN_TIME\n\tenum { IDD = IDD_TASKBAR_COLOR_DIALOG };\n#endif\n\nprotected:\n    std::map<CommonDisplayItem, TaskbarItemColor> m_colors;\n    CColorSettingListCtrl m_list_ctrl;\n\n    virtual CString GetDialogName() const override;\n\nprotected:\n\tvirtual void DoDataExchange(CDataExchange* pDX);    // DDX/DDV 支持\n\n\tDECLARE_MESSAGE_MAP()\npublic:\n\tvirtual BOOL OnInitDialog();\n    afx_msg void OnNMDblclkList1(NMHDR *pNMHDR, LRESULT *pResult);\n};\n"
  },
  {
    "path": "TrafficMonitor/TaskbarDefaultStyle.cpp",
    "content": "﻿#include \"stdafx.h\"\n#include \"TaskbarDefaultStyle.h\"\n#include \"IniHelper.h\"\n#include \"TrafficMonitor.h\"\n#include \"WindowsSettingHelper.h\"\n\nCTaskbarDefaultStyle::CTaskbarDefaultStyle()\n{\n}\n\nCTaskbarDefaultStyle::~CTaskbarDefaultStyle()\n{\n\tSaveConfig();\n}\n\nvoid CTaskbarDefaultStyle::LoadConfig()\n{\n\tCIniHelper ini{ theApp.m_config_path };\n\tfor (int i = 0; i < TASKBAR_DEFAULT_STYLE_NUM; i++)\n\t{\n        COLORREF default_text_color = (i == TASKBAR_DEFAULT_LIGHT_STYLE_INDEX ? RGB(0, 0, 0) : RGB(255, 255, 255));\n        COLORREF default_back_color = (i == TASKBAR_DEFAULT_LIGHT_STYLE_INDEX ? RGB(210, 210, 211) : 0);\n        COLORREF default_transparent_color = (i == TASKBAR_DEFAULT_LIGHT_STYLE_INDEX ? RGB(210, 210, 211) : 0);\n        COLORREF default_status_bar_color = (i == TASKBAR_DEFAULT_LIGHT_STYLE_INDEX ? RGB(165, 165, 165) : 0x005A5A5A);\n        wchar_t buff[64];\n        swprintf_s(buff, L\"default%d_\", i + 1);\n        wstring key_name = buff;\n        ini.LoadTaskbarWndColors(L\"taskbar_default_style\", (key_name + L\"text_color\").c_str(), m_default_style[i].text_colors, default_text_color);\n        m_default_style[i].back_color = ini.GetInt(L\"taskbar_default_style\", (key_name + L\"back_color\").c_str(), default_back_color);\n        m_default_style[i].transparent_color = ini.GetInt(L\"taskbar_default_style\", (key_name + L\"transparent_color\").c_str(), default_transparent_color);\n        m_default_style[i].status_bar_color = ini.GetInt(L\"taskbar_default_style\", (key_name + L\"status_bar_color\").c_str(), default_status_bar_color);\n        m_default_style[i].specify_each_item_color = ini.GetBool(L\"taskbar_default_style\", (key_name + L\"specify_each_item_color\").c_str(), false);\n\t}\n}\n\nvoid CTaskbarDefaultStyle::SaveConfig() const\n{\n\n\tCIniHelper ini{ theApp.m_config_path };\n\tfor (int i = 0; i < TASKBAR_DEFAULT_STYLE_NUM; i++)\n\t{\n\t\twchar_t buff[64];\n\t\tswprintf_s(buff, L\"default%d_\", i + 1);\n\t\twstring key_name = buff;\n        if (IsTaskBarStyleDataValid(m_default_style[i]))           //保存前检查当前颜色预设是否有效\n        {\n            ini.SaveTaskbarWndColors(L\"taskbar_default_style\", (key_name + L\"text_color\").c_str(), m_default_style[i].text_colors);\n            ini.WriteInt(L\"taskbar_default_style\", (key_name + L\"back_color\").c_str(), m_default_style[i].back_color);\n            ini.WriteInt(L\"taskbar_default_style\", (key_name + L\"transparent_color\").c_str(), m_default_style[i].transparent_color);\n            ini.WriteInt(L\"taskbar_default_style\", (key_name + L\"status_bar_color\").c_str(), m_default_style[i].status_bar_color);\n            ini.WriteBool(L\"taskbar_default_style\", (key_name + L\"specify_each_item_color\").c_str(), m_default_style[i].specify_each_item_color);\n        }\n        else\n        {\n            //写入日志\n            CString log_str;\n            log_str.Format(_T(\"在保存预设%d时检测到背景色和文字颜色都为黑色，该预设未被保存。\"), i);\n            CCommon::WriteLog(log_str, theApp.m_log_path.c_str());\n            return;\n        }\n\t}\n\tini.Save();\n}\n\nvoid CTaskbarDefaultStyle::ApplyDefaultStyle(int index, TaskBarSettingData & data) const\n{\n    /*if (index == TASKBAR_DEFAULT_LIGHT_STYLE_INDEX)\n    {\n        ApplyDefaultLightStyle(data);\n    }\n    else */if (index >= 0 && index < TASKBAR_DEFAULT_STYLE_NUM)\n\t{\n        if (!IsTaskBarStyleDataValid(m_default_style[index]))\n            return;\n\n        data.text_colors = m_default_style[index].text_colors;\n\t\tdata.back_color = m_default_style[index].back_color;\n\t\tdata.transparent_color = m_default_style[index].transparent_color;\n\t\tdata.status_bar_color = m_default_style[index].status_bar_color;\n\t\tdata.specify_each_item_color = m_default_style[index].specify_each_item_color;\n\t\tif (data.transparent_color == data.back_color)\n\t\t{\n\t\t\tCCommon::TransparentColorConvert(data.back_color);\n\t\t\tCCommon::TransparentColorConvert(data.transparent_color);\n\t\t}\n\t}\n}\n\n//void CTaskbarDefaultStyle::ApplyDefaultLightStyle(TaskBarSettingData& data)\n//{\n//\tfor (auto& item : data.text_colors)\n//    {\n//        item.second.label = RGB(0, 0, 0);\n//        item.second.value = RGB(0, 0, 0);\n//    }\n//\tdata.back_color = RGB(210, 210, 211);\n//\tdata.transparent_color = RGB(210, 210, 211);\n//\tdata.status_bar_color = RGB(165, 165, 165);\n//}\n\nvoid CTaskbarDefaultStyle::ModifyDefaultStyle(int index, TaskBarSettingData & data)\n{\n\tif (index < 0 || index >= TASKBAR_DEFAULT_STYLE_NUM)\n\t\treturn;\n\n    m_default_style[index].text_colors = data.text_colors;\n\tm_default_style[index].back_color = data.back_color;\n\tm_default_style[index].transparent_color = data.transparent_color;\n\tm_default_style[index].status_bar_color = data.status_bar_color;\n\tm_default_style[index].specify_each_item_color = data.specify_each_item_color;\n}\n\nbool CTaskbarDefaultStyle::IsTaskbarTransparent(const TaskBarSettingData& data)\n{\n\tif (CWindowsSettingHelper::IsWindows10LightTheme() || theApp.m_win_version.IsWindows8Or8point1() || theApp.m_is_windows11_taskbar)\n\t\treturn (data.transparent_color == data.back_color);\n\telse\n\t\treturn data.transparent_color == 0;\n}\n\nvoid CTaskbarDefaultStyle::SetTaskabrTransparent(bool transparent, TaskBarSettingData& data)\n{\n\tif (transparent)\n\t{\n\t\tif (CWindowsSettingHelper::IsWindows10LightTheme() || theApp.m_win_version.IsWindows8Or8point1() || theApp.m_is_windows11_taskbar)\n\t\t{\n\t\t\t//浅色模式下要设置任务栏窗口透明，只需将透明色设置成和背景色一样即可\n\t\t\tCCommon::TransparentColorConvert(data.back_color);\n\t\t\tdata.transparent_color = data.back_color;\n\t\t}\n\t\telse\n\t\t{\n\t\t\t//深色模式下，背景色透明将透明色设置成黑色\n\t\t\tdata.transparent_color = 0;\n\t\t}\n\t}\n\telse\n\t{\n\t\t//要设置任务栏窗口不透明，只需将透明色设置成和背景色不一样即可\n\t\tif (data.back_color != TASKBAR_TRANSPARENT_COLOR1)\n\t\t\tdata.transparent_color = TASKBAR_TRANSPARENT_COLOR1;\n\t\telse\n\t\t\tdata.transparent_color = TASKBAR_TRANSPARENT_COLOR2;\n\t}\n}\n\nbool CTaskbarDefaultStyle::IsTaskBarStyleDataValid(const TaskBarStyleData& data)\n{\n    for (const auto& item : data.text_colors)\n    {\n        if (item.second.label != data.back_color || item.second.value != data.back_color)\n            return true;\n    }\n    return false;     //如果文本颜色全部等于背景颜色，则该颜色预设无效\n}\n"
  },
  {
    "path": "TrafficMonitor/TaskbarDefaultStyle.h",
    "content": "﻿#pragma once\n#include \"CommonData.h\"\n\n#define TASKBAR_DEFAULT_STYLE_NUM 4\n#define TASKBAR_DEFAULT_LIGHT_STYLE_INDEX 3\n\nclass CTaskbarDefaultStyle\n{\npublic:\n\tstruct TaskBarStyleData\n\t{\n        std::map<CommonDisplayItem, TaskbarItemColor> text_colors{};\n\t\tCOLORREF back_color{};\n\t\tCOLORREF transparent_color{};\n\t\tCOLORREF status_bar_color{};\n\t\tbool specify_each_item_color{};\n\t};\n\n\tCTaskbarDefaultStyle();\n\t~CTaskbarDefaultStyle();\n\n\tvoid LoadConfig();\n\tvoid SaveConfig() const;\n\n\tvoid ApplyDefaultStyle(int index, TaskBarSettingData& data) const;\t\t//应用一个颜色预设\n\t//static void ApplyDefaultLightStyle(TaskBarSettingData& data);\t\t//应用默认的浅色模式预设\n\tvoid ModifyDefaultStyle(int index, TaskBarSettingData& data);     //将当前颜色设置保存到一个预设方案\n\n\tstatic bool IsTaskbarTransparent(const TaskBarSettingData& data);\n\tstatic void SetTaskabrTransparent(bool transparent, TaskBarSettingData& data);\n\n    static bool IsTaskBarStyleDataValid(const TaskBarStyleData& data);     //判断一个颜色模式是否有效\n\nprivate:\n\tTaskBarStyleData m_default_style[TASKBAR_DEFAULT_STYLE_NUM];    //预设样式\n\n};\n"
  },
  {
    "path": "TrafficMonitor/TaskbarItemOrderHelper.cpp",
    "content": "﻿#include \"stdafx.h\"\n#include \"TaskbarItemOrderHelper.h\"\n#include \"Common.h\"\n#include \"TrafficMonitor.h\"\n\nCommonDisplayItem::CommonDisplayItem(DisplayItem item)\n{\n    is_plugin = false;\n    item_type = item;\n    plugin_item = nullptr;\n}\n\nCommonDisplayItem::CommonDisplayItem(IPluginItem* item)\n{\n    is_plugin = true;\n    plugin_item = item;\n}\n\nbool CommonDisplayItem::operator<(const CommonDisplayItem& item) const\n{\n    if (is_plugin && !item.is_plugin)\n        return false;\n    else if (!is_plugin && item.is_plugin)\n        return true;\n    else if (!is_plugin)\n        return item_type < item.item_type;\n    else\n        return theApp.m_plugins.GetItemIndex(plugin_item) < theApp.m_plugins.GetItemIndex(item.plugin_item);\n}\n\nCString CommonDisplayItem::GetItemName() const\n{\n    CString item_name;\n    if (is_plugin)\n    {\n        if (plugin_item != nullptr)\n            item_name = plugin_item->GetItemName();\n    }\n    else\n    {\n        switch (item_type)\n        {\n        case TDI_UP:\n            item_name = CCommon::LoadText(IDS_UPLOAD);\n            break;\n        case TDI_DOWN:\n            item_name = CCommon::LoadText(IDS_DOWNLOAD);\n            break;\n        case TDI_TOTAL_SPEED:\n            item_name = CCommon::LoadText(IDS_TOTAL_NET_SPEED);\n            break;\n        case TDI_CPU:\n            item_name = CCommon::LoadText(IDS_CPU_USAGE);\n            break;\n        case TDI_MEMORY:\n            item_name = CCommon::LoadText(IDS_MEMORY_USAGE);\n            break;\n#ifndef WITHOUT_TEMPERATURE\n        case TDI_GPU_USAGE:\n            item_name = CCommon::LoadText(IDS_GPU_USAGE);\n            break;\n        case TDI_CPU_TEMP:\n            item_name = CCommon::LoadText(IDS_CPU_TEMPERATURE);\n            break;\n        case TDI_GPU_TEMP:\n            item_name = CCommon::LoadText(IDS_GPU_TEMPERATURE);\n            break;\n        case TDI_HDD_TEMP:\n            item_name = CCommon::LoadText(IDS_HDD_TEMPERATURE);\n            break;\n        case TDI_MAIN_BOARD_TEMP:\n            item_name = CCommon::LoadText(IDS_MAINBOARD_TEMPERATURE);\n            break;\n        case TDI_HDD_USAGE:\n            item_name = CCommon::LoadText(IDS_HDD_USAGE);\n            break;\n        case TDI_CPU_FREQ:\n            item_name = CCommon::LoadText(IDS_CPU_FREQ);\n            break;\n#endif\n        default:\n            break;\n        }\n    }\n    return item_name;\n}\n\nbool CommonDisplayItem::operator==(const CommonDisplayItem& item) const\n{\n    if (is_plugin != item.is_plugin)\n        return false;\n    else if (!is_plugin)\n        return item_type == item.item_type;\n    else\n        return plugin_item == item.plugin_item;\n}\n\n\n//////////////////////////////////////////////////////////////////////////////////\n//////////////////////////////////////////////////////////////////////////////////\n\nCTaskbarItemOrderHelper::CTaskbarItemOrderHelper(bool displayed_only)\n    : m_displayed_only(displayed_only)\n{\n}\n\nvoid CTaskbarItemOrderHelper::Init()\n{\n    for (const auto& item : theApp.m_plugins.AllDisplayItemsWithPlugins())\n    {\n        m_all_item_in_default_order.push_back(item);\n    }\n}\n\nstd::vector<CommonDisplayItem> CTaskbarItemOrderHelper::GetAllDisplayItemsWithOrder() const\n{\n    std::vector<CommonDisplayItem> items;\n    for (auto i : m_item_order)\n    {\n        if (i >= 0 && i < static_cast<int>(m_all_item_in_default_order.size()))\n        {\n            if (m_displayed_only && !IsItemDisplayed(m_all_item_in_default_order[i]))\n            {\n                continue;\n            }\n            items.push_back(m_all_item_in_default_order[i]);\n        }\n    }\n\n    return items;\n}\n\nvoid CTaskbarItemOrderHelper::FromString(const std::wstring& str)\n{\n    m_item_order.clear();\n    std::vector<std::wstring> str_list;\n    CCommon::StringSplit(str, L',', str_list);\n    for (const auto& s : str_list)\n    {\n        m_item_order.push_back(_wtoi(s.c_str()));\n    }\n    NormalizeItemOrder();\n}\n\nstd::wstring CTaskbarItemOrderHelper::ToString() const\n{\n    std::wstring result;\n    for (int i : m_item_order)\n    {\n        result += std::to_wstring(i);\n        result.push_back(L',');\n    }\n    if (!m_item_order.empty())\n        result.pop_back();\n    return result;\n}\n\nvoid CTaskbarItemOrderHelper::SetOrder(const vector<int>& item_order)\n{\n    m_item_order = item_order;\n    NormalizeItemOrder();\n}\n\nconst vector<int>& CTaskbarItemOrderHelper::GetItemOrderConst() const\n{\n    return m_item_order;\n}\n\nvector<int>& CTaskbarItemOrderHelper::GetItemOrder()\n{\n    return m_item_order;\n}\n\nCString CTaskbarItemOrderHelper::GetItemDisplayName(CommonDisplayItem item)\n{\n    if (item.is_plugin)\n    {\n        if (item.plugin_item != nullptr)\n            return item.plugin_item->GetItemName();\n        else\n            return CString();\n    }\n    else\n    {\n        switch (item.item_type)\n        {\n        case TDI_UP:\n            return CCommon::LoadText(IDS_UPLOAD);\n        case TDI_DOWN:\n            return CCommon::LoadText(IDS_DOWNLOAD);\n        case TDI_TOTAL_SPEED:\n            return CCommon::LoadText(IDS_TOTAL_NET_SPEED);\n        case TDI_CPU:\n            return CCommon::LoadText(IDS_CPU_USAGE);\n        case TDI_MEMORY:\n            return CCommon::LoadText(IDS_MEMORY_USAGE);\n        case TDI_GPU_USAGE:\n            return CCommon::LoadText(IDS_GPU_USAGE);\n        case TDI_CPU_TEMP:\n            return CCommon::LoadText(IDS_CPU_TEMPERATURE);\n        case TDI_GPU_TEMP:\n            return CCommon::LoadText(IDS_GPU_TEMPERATURE);\n        case TDI_HDD_TEMP:\n            return CCommon::LoadText(IDS_HDD_TEMPERATURE);\n        case TDI_MAIN_BOARD_TEMP:\n            return CCommon::LoadText(IDS_MAINBOARD_TEMPERATURE);\n        case TDI_HDD_USAGE:\n            return CCommon::LoadText(IDS_HDD_USAGE);\n        case TDI_CPU_FREQ:\n            return CCommon::LoadText(IDS_CPU_FREQ);\n        default:\n            break;\n        }\n    }\n    return CString();\n}\n\nbool CTaskbarItemOrderHelper::IsItemDisplayed(CommonDisplayItem item)\n{\n    bool displayed = true;\n    if (!item.is_plugin)\n    {\n        if ((item == TDI_CPU_TEMP || item == TDI_CPU_FREQ) && !theApp.m_general_data.IsHardwareEnable(HI_CPU))\n            displayed = false;\n        if ((item == TDI_GPU_TEMP || item == TDI_GPU_USAGE) && !theApp.m_general_data.IsHardwareEnable(HI_GPU))\n            displayed = false;\n        if ((item == TDI_HDD_TEMP || item == TDI_HDD_USAGE) && !theApp.m_general_data.IsHardwareEnable(HI_HDD))\n            displayed = false;\n        if (item == TDI_MAIN_BOARD_TEMP && !theApp.m_general_data.IsHardwareEnable(HI_MBD))\n            displayed = false;\n    }\n\n    return displayed;\n}\n\nvoid CTaskbarItemOrderHelper::NormalizeItemOrder()\n{\n    //检查是否有超出范围的序号\n    int item_num = static_cast<int>(theApp.m_plugins.AllDisplayItemsWithPlugins().size());\n    for (auto iter = m_item_order.begin(); iter != m_item_order.end();)\n    {\n        if (*iter < 0 || *iter >= item_num)\n            iter = m_item_order.erase(iter);\n        else\n            ++iter;\n    }\n    //删除不显示的序号\n    if (m_displayed_only)\n    {\n        for (auto iter = m_item_order.begin(); iter != m_item_order.end();)\n        {\n            if (*iter >= 0 && *iter < static_cast<int>(m_all_item_in_default_order.size()))\n            {\n                CommonDisplayItem item = m_all_item_in_default_order[*iter];\n                if (!IsItemDisplayed(item))\n                {\n                    iter = m_item_order.erase(iter);\n                    continue;\n                }\n            }\n            ++iter;\n        }\n    }\n    //删除重复的序号\n    CCommon::RemoveVectorDuplicateItem(m_item_order);\n    //检查是否有缺少的序号\n    for (int i = 0; i < item_num; i++)\n    {\n        auto iter = std::find(m_item_order.begin(), m_item_order.end(), i);\n        if (iter == m_item_order.end())\n            m_item_order.push_back(i);\n    }\n}\n"
  },
  {
    "path": "TrafficMonitor/TaskbarItemOrderHelper.h",
    "content": "﻿#pragma once\n#include \"PluginInterface.h\"\n\n//显示的项目\nenum DisplayItem\n{\n    TDI_UP = 1 << 0,\n    TDI_DOWN = 1 << 1,\n    TDI_CPU = 1 << 2,\n    TDI_MEMORY = 1 << 3,\n    TDI_GPU_USAGE = 1 << 4,\n    TDI_CPU_TEMP = 1 << 5,\n    TDI_GPU_TEMP = 1 << 6,\n    TDI_HDD_TEMP = 1 << 7,\n    TDI_MAIN_BOARD_TEMP = 1 << 8,\n    TDI_HDD_USAGE = 1 << 9,\n    TDI_TOTAL_SPEED = 1 << 10,\n    TDI_CPU_FREQ = 1 << 11\n};\n\n//显示的项目\nstruct CommonDisplayItem\n{\n    bool is_plugin{};           //是否为插件项目\n    DisplayItem item_type{};    //内建的显示项目\n    IPluginItem* plugin_item{}; //插件显示项目\n\n    CommonDisplayItem() {}\n    CommonDisplayItem(DisplayItem item);\n    CommonDisplayItem(IPluginItem* item);\n\n    bool operator<(const CommonDisplayItem&) const;\n    bool operator==(const CommonDisplayItem&) const;\n\n    CString GetItemName() const;\n};\n\n//所有显示项目的集合\nconst std::set<DisplayItem> AllDisplayItems\n{\n    TDI_UP, TDI_DOWN, TDI_CPU, TDI_MEMORY\n#ifndef WITHOUT_TEMPERATURE\n    , TDI_GPU_USAGE, TDI_CPU_TEMP, TDI_GPU_TEMP, TDI_HDD_TEMP, TDI_MAIN_BOARD_TEMP, TDI_HDD_USAGE,TDI_CPU_FREQ\n#endif\n    , TDI_TOTAL_SPEED\n};\n\n\nclass CTaskbarItemOrderHelper\n{\npublic:\n    CTaskbarItemOrderHelper(bool displayed_only = false);\n\n    void Init();\n\n    //根据设定的顺序获取任务窗口每个显示项目\n    std::vector<CommonDisplayItem> GetAllDisplayItemsWithOrder() const;\n\n    void FromString(const std::wstring& str);\n    std::wstring ToString() const;\n    void SetOrder(const vector<int>& item_order);\n    const vector<int>& GetItemOrderConst() const;\n    vector<int>& GetItemOrder();\n\n    static CString GetItemDisplayName(CommonDisplayItem item);\n    static bool IsItemDisplayed(CommonDisplayItem item);\n\nprivate:\n    //规范m_item_order里的项目，如果m_item_order里有序号超过了显示项目的个数，则将其移除，并在后面添加缺少的项目的序号\n    void NormalizeItemOrder();\n\nprivate:\n    vector<int> m_item_order;   //保存每个项目的顺序\n    vector<CommonDisplayItem> m_all_item_in_default_order;    //以默认顺序保存的所有显示项目\n    bool m_displayed_only;\n};\n"
  },
  {
    "path": "TrafficMonitor/Test.cpp",
    "content": "﻿#include \"stdafx.h\"\n#include \"Test.h\"\n#include \"Common.h\"\n#include \"SkinFile.h\"\n#include \"TrafficMonitor.h\"\n\nCTest::CTest()\n{\n}\n\nCTest::~CTest()\n{\n}\n\nvoid CTest::Test()\n{\n    //TestHttpQequest();\n    //TestGetLicense();\n    //TestSkin();\n    //TestCrash();\n    //TestDate();\n}\n\nvoid CTest::TestCommand()\n{\n    //TestPlugin();\n}\n\nvoid CTest::TestHttpQequest()\n{\n    wstring result;\n    bool rtn = CCommon::GetURL(L\"https://v4.yinghualuo.cn/bejson\", result, true, L\"TrafficMonitor_V1.78\");\n    int a = 0;\n}\n\nvoid CTest::TestGetLicense()\n{\n    CString license_str;\n    HRSRC hRes = FindResource(NULL, MAKEINTRESOURCE(IDR_LICENSE), _T(\"TEXT\"));\n    if (hRes != NULL)\n    {\n        HGLOBAL hglobal = LoadResource(NULL, hRes);\n        if (hglobal != NULL)\n        {\n            license_str = CCommon::StrToUnicode((const char*)hglobal, true).c_str();\n            int a = 0;\n        }\n    }\n}\n\nvoid CTest::TestSkin()\n{\n    CSkinFile skin;\n    //skin.Load(L\"C:\\\\Users\\\\zhong\\\\OneDrive\\\\文档\\\\TrafficMonitorSkin.xml\");\n    skin.Load(L\"D:\\\\Projects\\\\GitHub Project\\\\TrafficMonitor\\\\TrafficMonitor\\\\skins\\\\0默认皮肤\\\\skin.ini\");\n    int a = 0;\n}\n\nvoid CTest::TestCrash()\n{\n    CString* pStr = nullptr;\n    int a = pStr->GetLength();\n    printf(\"%d\", a);\n\n}\n\nvoid CTest::TestPlugin()\n{\n    if (!theApp.m_plugins.GetPlugins().empty())\n    {\n        theApp.m_plugins.GetPlugins()[0].plugin->ShowOptionsDialog(theApp.m_pMainWnd->m_hWnd);\n    }\n}\n\nvoid CTest::TestDate()\n{\n    Date d;\n    d.year = 2021;\n    d.month = 1;\n    d.day = 4;\n    int week = d.week();\n    int a = 0;\n}\n"
  },
  {
    "path": "TrafficMonitor/Test.h",
    "content": "﻿#pragma once\nclass CTest\n{\npublic:\n    CTest();\n    ~CTest();\n\n    static void Test();\n    static void TestCommand();\n\nprivate:\n    static void TestHttpQequest();\n    static void TestGetLicense();\n    static void TestSkin();\n    static void TestCrash();\n    static void TestPlugin();\n    static void TestDate();\n};\n"
  },
  {
    "path": "TrafficMonitor/TinyXml2Helper.cpp",
    "content": "﻿#include \"stdafx.h\"\n#include \"TinyXml2Helper.h\"\n#include \"Common.h\"\n\nbool CTinyXml2Helper::LoadXmlFile(tinyxml2::XMLDocument& doc, const wchar_t* file_path)\n{\n    //由于XMLDocument::LoadFile函数不支持Unicode，因此这里自行读取文件内容，并调用XMLDocument::Parse函数解析\n    size_t length;\n    const char* xml_contents = CCommon::GetFileContent(file_path, length);\n    auto err = doc.Parse(xml_contents, length);\n    delete[] xml_contents;\n    return err == tinyxml2::XML_SUCCESS;\n}\n\nvoid CTinyXml2Helper::IterateChildNode(tinyxml2::XMLElement* ele, std::function<void(tinyxml2::XMLElement*)> fun)\n{\n    if (ele == nullptr)\n        return;\n\n    tinyxml2::XMLElement* child = ele->FirstChildElement();\n    if (child == nullptr)\n        return;\n    fun(child);\n    while (true)\n    {\n        child = child->NextSiblingElement();\n        if (child != nullptr)\n            fun(child);\n        else\n            break;\n    }\n}\n\nconst char * CTinyXml2Helper::ElementAttribute(tinyxml2::XMLElement * ele, const char * attr)\n{\n    if (ele != nullptr)\n    {\n        const char* str = ele->Attribute(attr);\n        if (str != nullptr)\n            return str;\n    }\n    return \"\";\n}\n\nconst char* CTinyXml2Helper::ElementName(tinyxml2::XMLElement* ele)\n{\n    if (ele != nullptr)\n    {\n        const char* str = ele->Name();\n        if (str != nullptr)\n            return str;\n    }\n    return \"\";\n}\n\nconst char* CTinyXml2Helper::ElementText(tinyxml2::XMLElement* ele)\n{\n    if (ele != nullptr)\n    {\n        const char* str = ele->GetText();\n        if (str != nullptr)\n            return str;\n    }\n    return \"\";\n}\n\nbool CTinyXml2Helper::StringToBool(const char* str)\n{\n    string str_text{ str };\n    return (!str_text.empty() && str_text != \"0\");\n}\n\n"
  },
  {
    "path": "TrafficMonitor/TinyXml2Helper.h",
    "content": "﻿#pragma once\n#include \"tinyxml2/tinyxml2.h\"\n#include <string>\n#include <functional>\n\nclass CTinyXml2Helper\n{\npublic:\n\n    //从文件读取XML内容\n    static bool LoadXmlFile(tinyxml2::XMLDocument& doc, const wchar_t* file_path);\n\n    //遍历一个XML节点\n    //fun: 一个函数对象，遍历到一个节点时被调用\n    static void IterateChildNode(tinyxml2::XMLElement* ele, std::function<void(tinyxml2::XMLElement*)> fun);\n\n    //获取一个节点的属性（返回值不会为空指针，如果找不到则返回空字符串）\n    static const char* ElementAttribute(tinyxml2::XMLElement* ele, const char* attr);\n\n    //获取一个节点的名称（返回值不会为空指针，如果找不到则返回空字符串）\n    static const char* ElementName(tinyxml2::XMLElement* ele);\n\n    //获取一个节点的文本（返回值不会为空指针，如果找不到则返回空字符串）\n    static const char* ElementText(tinyxml2::XMLElement* ele);\n\n    static bool StringToBool(const char* str);\n};\n"
  },
  {
    "path": "TrafficMonitor/TrafficMonitor.cpp",
    "content": "﻿\n// TrafficMonitor.cpp : 定义应用程序的类行为。\n//\n\n#include \"stdafx.h\"\n#include \"TrafficMonitor.h\"\n#include \"TrafficMonitorDlg.h\"\n#include \"crashtool.h\"\n#include \"UpdateHelper.h\"\n#include \"Test.h\"\n#include \"WIC.h\"\n#include \"auto_start_helper.h\"\n#include \"AppAlreadyRuningDlg.h\"\n#include \"WindowsSettingHelper.h\"\n\n#ifdef _DEBUG\n#define new DEBUG_NEW\n#endif\n\n\n// CTrafficMonitorApp\n\nBEGIN_MESSAGE_MAP(CTrafficMonitorApp, CWinApp)\n    //ON_COMMAND(ID_HELP, &CWinApp::OnHelp)\n    ON_COMMAND(ID_HELP, &CTrafficMonitorApp::OnHelp)\n    ON_COMMAND(ID_FREQUENTY_ASKED_QUESTIONS, &CTrafficMonitorApp::OnFrequentyAskedQuestions)\n    ON_COMMAND(ID_UPDATE_LOG, &CTrafficMonitorApp::OnUpdateLog)\nEND_MESSAGE_MAP()\n\n\nCTrafficMonitorApp* CTrafficMonitorApp::self = NULL;\n\n\n// CTrafficMonitorApp 构造\nCTrafficMonitorApp::CTrafficMonitorApp()\n{\n    self = this;\n    // 支持重新启动管理器\n    //m_dwRestartManagerSupportFlags = AFX_RESTART_MANAGER_SUPPORT_RESTART;\n\n    // TODO: 在此处添加构造代码，\n    // 将所有重要的初始化放置在 InitInstance 中\n    CRASHREPORT::StartCrashReport();\n}\n\nvoid CTrafficMonitorApp::LoadConfig()\n{\n    CIniHelper ini{ m_config_path };\n\n    //常规设置\n    m_general_data.check_update_when_start = ini.GetBool(_T(\"general\"), _T(\"check_update_when_start\"), true);\n    m_general_data.allow_skin_cover_font = ini.GetBool(_T(\"general\"), _T(\"allow_skin_cover_font\"), true);\n    m_general_data.allow_skin_cover_text = ini.GetBool(_T(\"general\"), _T(\"allow_skin_cover_text\"), true);\n    m_general_data.language = static_cast<Language>(ini.GetInt(_T(\"general\"), _T(\"language\"), 0));\n    m_general_data.show_all_interface = ini.GetBool(L\"general\", L\"show_all_interface\", false);\n    bool is_chinese_language{};     //当前语言是否为简体中文\n    if (m_general_data.language == Language::FOLLOWING_SYSTEM)\n        is_chinese_language = CCommon::LoadText(IDS_LANGUAGE_CODE) == _T(\"2\");\n    else\n        is_chinese_language = (m_general_data.language == Language::SIMPLIFIED_CHINESE);\n    m_general_data.update_source = ini.GetInt(L\"general\", L\"update_source\", is_chinese_language ? 1 : 0);   //如果当前语言为简体，则默认更新源为Gitee，否则为GitHub\n    //载入获取CPU利用率的方式，默认使用GetSystemTimes获取\n    m_general_data.m_get_cpu_usage_by_cpu_times = ini.GetBool(L\"general\", L\"get_cpu_usage_by_cpu_times\", /*m_win_version.GetMajorVersion() < 10*/ true);\n    m_general_data.monitor_time_span = ini.GetInt(L\"general\", L\"monitor_time_span\", 1000);\n    if (m_general_data.monitor_time_span < MONITOR_TIME_SPAN_MIN || m_general_data.monitor_time_span > MONITOR_TIME_SPAN_MAX)\n        m_general_data.monitor_time_span = 1000;\n    m_general_data.hard_disk_name = ini.GetString(L\"general\", L\"hard_disk_name\", L\"\");\n    m_general_data.cpu_core_name = ini.GetString(L\"general\", L\"cpu_core_name\", L\"Core Average\");\n    m_general_data.hardware_monitor_item = ini.GetInt(L\"general\", L\"hardware_monitor_item\", 0);\n    std::vector<std::wstring> connections_hide;\n    ini.GetStringList(L\"general\", L\"connections_hide\", connections_hide, std::vector<std::wstring>{});\n    m_general_data.connections_hide.FromVector(connections_hide);\n\n    //Windows10颜色模式设置\n    bool is_windows10_light_theme = CWindowsSettingHelper::IsWindows10LightTheme();\n    if (is_windows10_light_theme)\n        CCommon::SetColorMode(ColorMode::Light);\n    else\n        CCommon::SetColorMode(ColorMode::Default);\n\n    //主窗口设置\n    m_cfg_data.m_transparency = ini.GetInt(_T(\"config\"), _T(\"transparency\"), 80);\n    m_main_wnd_data.m_always_on_top = ini.GetBool(_T(\"config\"), _T(\"always_on_top\"), true);\n    m_main_wnd_data.m_lock_window_pos = ini.GetBool(_T(\"config\"), _T(\"lock_window_pos\"), false);\n    m_general_data.show_notify_icon = ini.GetBool(_T(\"config\"), _T(\"show_notify_icon\"), true);\n    m_cfg_data.m_show_more_info = ini.GetBool(_T(\"config\"), _T(\"show_cpu_memory\"), false);\n    m_main_wnd_data.m_mouse_penetrate = ini.GetBool(_T(\"config\"), _T(\"mouse_penetrate\"), false);\n    m_cfg_data.m_show_task_bar_wnd = ini.GetBool(_T(\"config\"), _T(\"show_task_bar_wnd\"), false);\n    m_cfg_data.m_position_x = ini.GetInt(_T(\"config\"), _T(\"position_x\"), -1);\n    m_cfg_data.m_position_y = ini.GetInt(_T(\"config\"), _T(\"position_y\"), -1);\n    m_cfg_data.m_auto_select = ini.GetBool(_T(\"connection\"), _T(\"auto_select\"), true);\n    m_cfg_data.m_select_all = ini.GetBool(_T(\"connection\"), _T(\"select_all\"), false);\n    //判断皮肤是否存在\n    std::vector<wstring> skin_files;\n    CCommon::GetFiles((theApp.m_skin_path + L\"\\\\*\").c_str(), skin_files);\n    bool is_skin_exist = (!skin_files.empty());\n    ini.LoadMainWndColors(_T(\"config\"), _T(\"text_color\"), m_main_wnd_data.text_colors, (is_skin_exist ? 16384 : 16777215)); //根据皮肤是否存在来设置默认的文本颜色，皮肤文件不存在时文本颜色默认为白色\n    m_main_wnd_data.specify_each_item_color = ini.GetBool(_T(\"config\"), _T(\"specify_each_item_color\"), false);\n    m_cfg_data.m_hide_main_window = ini.GetBool(_T(\"config\"), _T(\"hide_main_window\"), false);\n    m_cfg_data.m_connection_name = CCommon::UnicodeToStr(ini.GetString(L\"connection\", L\"connection_name\", L\"\").c_str());\n    m_cfg_data.m_skin_name = ini.GetString(_T(\"config\"), _T(\"skin_selected\"), _T(\"\"));\n    if (m_cfg_data.m_skin_name.substr(0, 8) == L\".\\\\skins\\\\\")       //如果读取到的皮肤名称前面有\".\\\\skins\\\\\"，则把它删除。（用于和前一个版本保持兼容性）\n        m_cfg_data.m_skin_name = m_cfg_data.m_skin_name.substr(7);\n    m_cfg_data.m_notify_icon_selected = ini.GetInt(_T(\"config\"), _T(\"notify_icon_selected\"), (m_win_version.IsWindows7() || m_win_version.IsWindows8Or8point1() ? 2 : m_cfg_data.m_dft_notify_icon));       //Win7/8/8.1默认使用蓝色通知区图标，因为隐藏通知区图标后白色图标会看不清，其他系统默认使用白色图标\n    m_cfg_data.m_notify_icon_auto_adapt = ini.GetBool(_T(\"config\"), _T(\"notify_icon_auto_adapt\"), true);\n    if (m_cfg_data.m_notify_icon_auto_adapt)\n        AutoSelectNotifyIcon();\n    m_main_wnd_data.swap_up_down = ini.GetBool(_T(\"config\"), _T(\"swap_up_down\"), false);\n    m_main_wnd_data.hide_main_wnd_when_fullscreen = ini.GetBool(_T(\"config\"), _T(\"hide_main_wnd_when_fullscreen\"), true);\n\n    FontInfo default_font{};\n    default_font.name = CCommon::LoadText(IDS_DEFAULT_FONT);\n    default_font.size = 10;\n    ini.LoadFontData(_T(\"config\"), m_main_wnd_data.font, default_font);\n    //m_main_wnd_data.font.name = ini.GetString(_T(\"config\"), _T(\"font_name\"), CCommon::LoadText(IDS_MICROSOFT_YAHEI)).c_str();\n    //m_main_wnd_data.font.size = ini.GetInt(_T(\"config\"), _T(\"font_size\"), 10);\n\n    //载入显示文本设置\n    m_main_wnd_data.disp_str.Get(TDI_UP) = ini.GetString(_T(\"config\"), L\"up_string\", CCommon::LoadText(IDS_UPLOAD_DISP, _T(\": $\")));\n    m_main_wnd_data.disp_str.Get(TDI_DOWN) = ini.GetString(L\"config\", L\"down_string\", CCommon::LoadText(IDS_DOWNLOAD_DISP, _T(\": $\")));\n    m_main_wnd_data.disp_str.Get(TDI_TOTAL_SPEED) = ini.GetString(L\"config\", L\"total_speed_string\", _T(\"↑↓: $\"));\n    m_main_wnd_data.disp_str.Get(TDI_CPU) = ini.GetString(L\"config\", L\"cpu_string\", L\"CPU: $\");\n    m_main_wnd_data.disp_str.Get(TDI_CPU_FREQ) = ini.GetString(L\"config\", L\"cpu_freq_string\", CCommon::LoadText(IDS_CPU_FREQ, _T(\": $\")));\n    m_main_wnd_data.disp_str.Get(TDI_MEMORY) = ini.GetString(L\"config\", L\"memory_string\", CCommon::LoadText(IDS_MEMORY_DISP, _T(\": $\")));\n    m_main_wnd_data.disp_str.Get(TDI_GPU_USAGE) = ini.GetString(L\"config\", L\"gpu_string\", CCommon::LoadText(IDS_GPU_DISP, _T(\": $\")));\n    m_main_wnd_data.disp_str.Get(TDI_CPU_TEMP) = ini.GetString(L\"config\", L\"cpu_temp_string\", L\"CPU: $\");\n    m_main_wnd_data.disp_str.Get(TDI_GPU_TEMP) = ini.GetString(L\"config\", L\"gpu_temp_string\", CCommon::LoadText(IDS_GPU_DISP, _T(\": $\")));\n    m_main_wnd_data.disp_str.Get(TDI_HDD_TEMP) = ini.GetString(L\"config\", L\"hdd_temp_string\", CCommon::LoadText(IDS_HDD_DISP, _T(\": $\")));\n    m_main_wnd_data.disp_str.Get(TDI_MAIN_BOARD_TEMP) = ini.GetString(L\"config\", L\"main_board_temp_string\", CCommon::LoadText(IDS_MAINBOARD_DISP, _T(\": $\")));\n    m_main_wnd_data.disp_str.Get(TDI_HDD_USAGE) = ini.GetString(L\"config\", L\"hdd_string\", CCommon::LoadText(IDS_HDD_DISP, _T(\": $\")));\n\n    //载入插件项目的显示文本设置\n    ini.LoadPluginDisplayStr(true);\n\n    m_main_wnd_data.speed_short_mode = ini.GetBool(_T(\"config\"), _T(\"speed_short_mode\"), false);\n    m_main_wnd_data.separate_value_unit_with_space = ini.GetBool(_T(\"config\"), _T(\"separate_value_unit_with_space\"), true);\n    m_main_wnd_data.show_tool_tip = ini.GetBool(_T(\"config\"), _T(\"show_tool_tip\"), true);\n    m_main_wnd_data.memory_display = static_cast<MemoryDisplay>(ini.GetInt(L\"config\", L\"memory_display\", static_cast<int>(MemoryDisplay::USAGE_PERCENTAGE)));\n    m_main_wnd_data.unit_byte = ini.GetBool(_T(\"config\"), _T(\"unit_byte\"), true);\n    m_main_wnd_data.speed_unit = static_cast<SpeedUnit>(ini.GetInt(_T(\"config\"), _T(\"speed_unit\"), 0));\n    m_main_wnd_data.hide_unit = ini.GetBool(_T(\"config\"), _T(\"hide_unit\"), false);\n    m_main_wnd_data.hide_percent = ini.GetBool(_T(\"config\"), _T(\"hide_percent\"), false);\n    m_main_wnd_data.double_click_action = static_cast<DoubleClickAction>(ini.GetInt(_T(\"config\"), _T(\"double_click_action\"), 0));\n    m_main_wnd_data.double_click_exe = ini.GetString(L\"config\", L\"double_click_exe\", (theApp.m_system_dir + L\"\\\\Taskmgr.exe\").c_str());\n\n    m_main_wnd_data.m_alow_out_of_border = ini.GetBool(_T(\"config\"), _T(\"alow_out_of_border\"), false);\n\n    m_general_data.traffic_tip_enable = ini.GetBool(L\"notify_tip\", L\"traffic_tip_enable\", false);\n    m_general_data.traffic_tip_value = ini.GetInt(L\"notify_tip\", L\"traffic_tip_value\", 200);\n    m_general_data.traffic_tip_unit = ini.GetInt(L\"notify_tip\", L\"traffic_tip_unit\", 0);\n    m_general_data.memory_usage_tip.enable = ini.GetBool(L\"notify_tip\", L\"memory_usage_tip_enable\", false);\n    m_general_data.memory_usage_tip.tip_value = ini.GetInt(L\"notify_tip\", L\"memory_tip_value\", 80);\n    m_general_data.cpu_temp_tip.enable = ini.GetBool(L\"notify_tip\", L\"cpu_temperature_tip_enable\", false);\n    m_general_data.cpu_temp_tip.tip_value = ini.GetInt(L\"notify_tip\", L\"cpu_temperature_tip_value\", 80);\n    m_general_data.gpu_temp_tip.enable = ini.GetBool(L\"notify_tip\", L\"gpu_temperature_tip_enable\", false);\n    m_general_data.gpu_temp_tip.tip_value = ini.GetInt(L\"notify_tip\", L\"gpu_temperature_tip_value\", 80);\n    m_general_data.hdd_temp_tip.enable = ini.GetBool(L\"notify_tip\", L\"hdd_temperature_tip_enable\", false);\n    m_general_data.hdd_temp_tip.tip_value = ini.GetInt(L\"notify_tip\", L\"hdd_temperature_tip_value\", 80);\n    m_general_data.mainboard_temp_tip.enable = ini.GetBool(L\"notify_tip\", L\"mainboard_temperature_tip_enable\", false);\n    m_general_data.mainboard_temp_tip.tip_value = ini.GetInt(L\"notify_tip\", L\"mainboard_temperature_tip_value\", 80);\n \n    //任务栏窗口设置\n    m_taskbar_data.back_color = ini.GetInt(_T(\"task_bar\"), _T(\"task_bar_back_color\"), m_taskbar_data.dft_back_color);\n    m_taskbar_data.transparent_color = ini.GetInt(_T(\"task_bar\"), _T(\"transparent_color\"), m_taskbar_data.dft_transparent_color);\n    if (CTaskbarDefaultStyle::IsTaskbarTransparent(m_taskbar_data)) //如果任务栏背景透明，则需要将颜色转换一下\n    {\n        CCommon::TransparentColorConvert(m_taskbar_data.back_color);\n        CCommon::TransparentColorConvert(m_taskbar_data.transparent_color);\n    }\n    m_taskbar_data.status_bar_color = ini.GetInt(_T(\"task_bar\"), _T(\"status_bar_color\"), m_taskbar_data.dft_status_bar_color);\n    //m_taskbar_data.text_color = GetPrivateProfileInt(_T(\"task_bar\"), _T(\"task_bar_text_color\"), 0x00ffffffU, m_config_path.c_str());\n    ini.LoadTaskbarWndColors(_T(\"task_bar\"), _T(\"task_bar_text_color\"), m_taskbar_data.text_colors, m_taskbar_data.dft_text_colors);\n    m_taskbar_data.specify_each_item_color = ini.GetBool(L\"task_bar\", L\"specify_each_item_color\", false);\n    //m_cfg_data.m_tbar_show_cpu_memory = ini.GetBool(_T(\"task_bar\"), _T(\"task_bar_show_cpu_memory\"), false);\n    m_taskbar_data.m_tbar_display_item = ini.GetInt(L\"task_bar\", L\"tbar_display_item\", TDI_UP | TDI_DOWN);\n\n    //不含温度监控的版本，不显示温度监控相关项目\n#ifdef WITHOUT_TEMPERATURE\n    m_taskbar_data.m_tbar_display_item &= ~TDI_GPU_USAGE;\n    m_taskbar_data.m_tbar_display_item &= ~TDI_CPU_TEMP;\n    m_taskbar_data.m_tbar_display_item &= ~TDI_GPU_TEMP;\n    m_taskbar_data.m_tbar_display_item &= ~TDI_HDD_TEMP;\n    m_taskbar_data.m_tbar_display_item &= ~TDI_MAIN_BOARD_TEMP;\n    m_taskbar_data.m_tbar_display_item &= ~TDI_HDD_USAGE;\n#endif\n\n    //如果选项设置中关闭了某个硬件监控，则不显示对应的温度监控相关项目\n    if (!m_general_data.IsHardwareEnable(HI_CPU))\n        m_taskbar_data.m_tbar_display_item &= ~TDI_CPU_TEMP;\n    if (!m_general_data.IsHardwareEnable(HI_GPU))\n    {\n        m_taskbar_data.m_tbar_display_item &= ~TDI_GPU_USAGE;\n        m_taskbar_data.m_tbar_display_item &= ~TDI_GPU_TEMP;\n    }\n    if (!m_general_data.IsHardwareEnable(HI_HDD))\n    {\n        m_taskbar_data.m_tbar_display_item &= ~TDI_HDD_TEMP;\n        m_taskbar_data.m_tbar_display_item &= ~TDI_HDD_USAGE;\n    }\n    if (!m_general_data.IsHardwareEnable(HI_MBD))\n        m_taskbar_data.m_tbar_display_item &= ~TDI_MAIN_BOARD_TEMP;\n\n    //m_taskbar_data.swap_up_down = ini.GetBool(_T(\"task_bar\"), _T(\"task_bar_swap_up_down\"), false);\n\n    if (m_taskbar_data.back_color == 0 && !m_taskbar_data.text_colors.empty() && m_taskbar_data.text_colors.begin()->second.label == 0)     //万一读取到的背景色和文本颜色都为0（黑色），则将文本色和背景色设置成默认颜色\n    {\n        m_taskbar_data.back_color = m_taskbar_data.dft_back_color;\n        m_taskbar_data.text_colors.begin()->second.label = m_taskbar_data.dft_text_colors;\n    }\n\n    //m_taskbar_data.font.name = ini.GetString(_T(\"task_bar\"), _T(\"tack_bar_font_name\"), CCommon::LoadText(IDS_MICROSOFT_YAHEI)).c_str();\n    //m_taskbar_data.font.size = ini.GetInt(_T(\"task_bar\"), _T(\"tack_bar_font_size\"), 9);\n    default_font = FontInfo{};\n    default_font.name = CCommon::LoadText(IDS_DEFAULT_FONT);\n    default_font.size = 9;\n    ini.LoadFontData(_T(\"task_bar\"), m_taskbar_data.font, default_font);\n\n    m_taskbar_data.disp_str.Get(TDI_UP) = ini.GetString(L\"task_bar\", L\"up_string\", L\"↑: $\");\n    m_taskbar_data.disp_str.Get(TDI_DOWN) = ini.GetString(L\"task_bar\", L\"down_string\", L\"↓: $\");\n    m_taskbar_data.disp_str.Get(TDI_TOTAL_SPEED) = ini.GetString(L\"task_bar\", L\"total_speed_string\", L\"↑↓: $\");\n    m_taskbar_data.disp_str.Get(TDI_CPU) = ini.GetString(L\"task_bar\", L\"cpu_string\", L\"CPU: $\");\n    m_taskbar_data.disp_str.Get(TDI_MEMORY) = ini.GetString(L\"task_bar\", L\"memory_string\", CCommon::LoadText(IDS_MEMORY_DISP, _T(\": $\")));\n    m_taskbar_data.disp_str.Get(TDI_GPU_USAGE) = ini.GetString(L\"task_bar\", L\"gpu_string\", CCommon::LoadText(IDS_GPU_DISP, _T(\": $\")));\n    m_taskbar_data.disp_str.Get(TDI_CPU_TEMP) = ini.GetString(L\"task_bar\", L\"cpu_temp_string\", L\"CPU: $\");\n    m_taskbar_data.disp_str.Get(TDI_GPU_TEMP) = ini.GetString(L\"task_bar\", L\"gpu_temp_string\", CCommon::LoadText(IDS_GPU_DISP, _T(\": \")));\n    m_taskbar_data.disp_str.Get(TDI_HDD_TEMP) = ini.GetString(L\"task_bar\", L\"hdd_temp_string\", CCommon::LoadText(IDS_HDD_DISP, _T(\": \")));\n    m_taskbar_data.disp_str.Get(TDI_MAIN_BOARD_TEMP) = ini.GetString(L\"task_bar\", L\"main_board_temp_string\", CCommon::LoadText(IDS_MAINBOARD_DISP, _T(\": \")));\n    m_taskbar_data.disp_str.Get(TDI_HDD_USAGE) = ini.GetString(L\"task_bar\", L\"hdd_string\", CCommon::LoadText(IDS_HDD_DISP, _T(\": \")));\n    m_taskbar_data.disp_str.Get(TDI_CPU_FREQ) = ini.GetString(L\"task_bar\", L\"cpu_freq_string\", CCommon::LoadText(IDS_CPU_FREQ, _T(\": $\")));\n    ini.LoadPluginDisplayStr(false);\n\n    m_taskbar_data.tbar_wnd_on_left = ini.GetBool(_T(\"task_bar\"), _T(\"task_bar_wnd_on_left\"), false);\n    m_taskbar_data.speed_short_mode = ini.GetBool(_T(\"task_bar\"), _T(\"task_bar_speed_short_mode\"), false);\n    m_taskbar_data.tbar_wnd_snap = ini.GetBool(_T(\"task_bar\"), _T(\"task_bar_wnd_snap\"), false);\n    m_taskbar_data.unit_byte = ini.GetBool(_T(\"task_bar\"), _T(\"unit_byte\"), true);\n    m_taskbar_data.speed_unit = static_cast<SpeedUnit>(ini.GetInt(_T(\"task_bar\"), _T(\"task_bar_speed_unit\"), 0));\n    m_taskbar_data.hide_unit = ini.GetBool(_T(\"task_bar\"), _T(\"task_bar_hide_unit\"), false);\n    m_taskbar_data.hide_percent = ini.GetBool(_T(\"task_bar\"), _T(\"task_bar_hide_percent\"), false);\n    m_taskbar_data.value_right_align = ini.GetBool(_T(\"task_bar\"), _T(\"value_right_align\"), false);\n    m_taskbar_data.horizontal_arrange = ini.GetBool(_T(\"task_bar\"), _T(\"horizontal_arrange\"), false);\n    m_taskbar_data.show_status_bar = ini.GetBool(_T(\"task_bar\"), _T(\"show_status_bar\"), false);\n    m_taskbar_data.separate_value_unit_with_space = ini.GetBool(_T(\"task_bar\"), _T(\"separate_value_unit_with_space\"), true);\n    m_taskbar_data.show_tool_tip = ini.GetBool(_T(\"task_bar\"), _T(\"show_tool_tip\"), true);\n    m_taskbar_data.digits_number = ini.GetInt(_T(\"task_bar\"), _T(\"digits_number\"), 4);\n    m_taskbar_data.memory_display = static_cast<MemoryDisplay>(ini.GetInt(L\"task_bar\", L\"memory_display\", static_cast<int>(MemoryDisplay::USAGE_PERCENTAGE)));\n    m_taskbar_data.double_click_action = static_cast<DoubleClickAction>(ini.GetInt(_T(\"task_bar\"), _T(\"double_click_action\"), 0));\n    m_taskbar_data.double_click_exe = ini.GetString(L\"task_bar\", L\"double_click_exe\", (theApp.m_system_dir + L\"\\\\Taskmgr.exe\").c_str());\n    m_taskbar_data.cm_graph_type = ini.GetBool(_T(\"task_bar\"), _T(\"cm_graph_type\"), true);\n    m_taskbar_data.show_graph_dashed_box = ini.GetBool(L\"task_bar\", L\"show_graph_dashed_box\", false);\n    m_taskbar_data.item_space = ini.GetInt(L\"task_bar\", L\"item_space\", 4);\n    m_taskbar_data.ValidItemSpace();\n\n    if (m_win_version.IsWindows10OrLater())     //只有Win10才支持自动适应系统深色/浅色主题\n        m_taskbar_data.auto_adapt_light_theme = ini.GetBool(L\"task_bar\", L\"auto_adapt_light_theme\", false);\n    else\n        m_taskbar_data.auto_adapt_light_theme = false;\n    m_taskbar_data.dark_default_style = ini.GetInt(L\"task_bar\", L\"dark_default_style\", 0);\n    m_taskbar_data.light_default_style = ini.GetInt(L\"task_bar\", L\"light_default_style\", TASKBAR_DEFAULT_LIGHT_STYLE_INDEX);\n\n    if (m_win_version.IsWindows8OrLater())\n        m_taskbar_data.auto_set_background_color = ini.GetBool(L\"task_bar\", L\"auto_set_background_color\", false);\n    else\n        m_taskbar_data.auto_set_background_color = false;\n\n    m_taskbar_data.item_order.Init();\n    m_taskbar_data.item_order.FromString(ini.GetString(L\"task_bar\", L\"item_order\", L\"\"));\n    m_taskbar_data.plugin_display_item.FromString(ini.GetString(L\"task_bar\", L\"plugin_display_item\", L\"\"));\n    m_taskbar_data.auto_save_taskbar_color_settings_to_preset = ini.GetBool(L\"task_bar\", L\"auto_save_taskbar_color_settings_to_preset\", true);\n\n    m_taskbar_data.show_netspeed_figure = ini.GetBool(L\"task_bar\", L\"show_netspeed_figure\", false);\n    m_taskbar_data.netspeed_figure_max_value = ini.GetInt(L\"task_bar\", L\"netspeed_figure_max_value\", 512);\n    m_taskbar_data.netspeed_figure_max_value_unit = ini.GetInt(L\"task_bar\", L\"netspeed_figure_max_value_unit\", 0);\n\n    //其他设置\n    //m_cfg_data.m_show_internet_ip = ini.GetBool(L\"connection_details\", L\"show_internet_ip\", false);\n    m_cfg_data.m_use_log_scale = ini.GetBool(_T(\"histroy_traffic\"), _T(\"use_log_scale\"), true);\n    m_cfg_data.m_sunday_first = ini.GetBool(_T(\"histroy_traffic\"), _T(\"sunday_first\"), true);\n    m_cfg_data.m_view_type = static_cast<HistoryTrafficViewType>(ini.GetInt(_T(\"histroy_traffic\"), _T(\"view_type\"), static_cast<int>(HistoryTrafficViewType::HV_DAY)));\n\n    m_no_multistart_warning = ini.GetBool(_T(\"other\"), _T(\"no_multistart_warning\"), false);\n    m_notify_interval = ini.GetInt(_T(\"other\"), _T(\"notify_interval\"), 60);\n    m_exit_when_start_by_restart_manager = ini.GetBool(_T(\"other\"), _T(\"exit_when_start_by_restart_manager\"), true);\n    m_debug_log = ini.GetBool(_T(\"other\"), _T(\"debug_log\"), false);\n    //由于Win7系统中设置任务栏窗口透明色会导致任务栏窗口不可见，因此默认在Win7中禁用透明色的设定\n    m_taksbar_transparent_color_enable = ini.GetBool(L\"other\", L\"taksbar_transparent_color_enable\", !m_win_version.IsWindows7());\n    m_last_light_mode = ini.GetBool(L\"other\", L\"last_light_mode\", CWindowsSettingHelper::IsWindows10LightTheme());\n    m_show_mouse_panetrate_tip = ini.GetBool(L\"other\", L\"show_mouse_panetrate_tip\", true);\n    m_show_dot_net_notinstalled_tip = ini.GetBool(L\"other\", L\"show_dot_net_notinstalled_tip\", true);\n\n    m_cfg_data.taskbar_left_space_win11 = ini.GetInt(L\"task_bar\", L\"taskbar_left_space_win11\", 160);\n}\n\nvoid CTrafficMonitorApp::SaveConfig()\n{\n    CIniHelper ini{ m_config_path };\n\n    //常规设置\n    ini.WriteBool(_T(\"general\"), _T(\"check_update_when_start\"), m_general_data.check_update_when_start);\n    ini.WriteBool(_T(\"general\"), _T(\"allow_skin_cover_font\"), m_general_data.allow_skin_cover_font);\n    ini.WriteBool(_T(\"general\"), _T(\"allow_skin_cover_text\"), m_general_data.allow_skin_cover_text);\n    ini.WriteInt(_T(\"general\"), _T(\"language\"), static_cast<int>(m_general_data.language));\n    ini.WriteBool(L\"general\", L\"show_all_interface\", m_general_data.show_all_interface);\n    ini.WriteBool(L\"general\", L\"get_cpu_usage_by_cpu_times\", m_general_data.m_get_cpu_usage_by_cpu_times);\n    ini.WriteInt(L\"general\", L\"monitor_time_span\", m_general_data.monitor_time_span);\n    ini.WriteString(L\"general\", L\"hard_disk_name\", m_general_data.hard_disk_name);\n    ini.WriteString(L\"general\", L\"cpu_core_name\", m_general_data.cpu_core_name);\n    ini.WriteInt(L\"general\", L\"hardware_monitor_item\", m_general_data.hardware_monitor_item);\n    ini.WriteStringList(L\"general\", L\"connections_hide\", m_general_data.connections_hide.ToVector());\n\n    //主窗口设置\n    ini.WriteInt(L\"config\", L\"transparency\", m_cfg_data.m_transparency);\n    ini.WriteBool(L\"config\", L\"always_on_top\", m_main_wnd_data.m_always_on_top);\n    ini.WriteBool(L\"config\", L\"lock_window_pos\", m_main_wnd_data.m_lock_window_pos);\n    ini.WriteBool(L\"config\", L\"show_notify_icon\", m_general_data.show_notify_icon);\n    ini.WriteBool(L\"config\", L\"show_cpu_memory\", m_cfg_data.m_show_more_info);\n    ini.WriteBool(L\"config\", L\"mouse_penetrate\", m_main_wnd_data.m_mouse_penetrate);\n    ini.WriteBool(L\"config\", L\"show_task_bar_wnd\", m_cfg_data.m_show_task_bar_wnd);\n    ini.WriteInt(L\"config\", L\"position_x\", m_cfg_data.m_position_x);\n    ini.WriteInt(L\"config\", L\"position_y\", m_cfg_data.m_position_y);\n    ini.WriteBool(L\"connection\", L\"auto_select\", m_cfg_data.m_auto_select);\n    ini.WriteBool(L\"connection\", L\"select_all\", m_cfg_data.m_select_all);\n    ini.SaveMainWndColors(L\"config\", L\"text_color\", m_main_wnd_data.text_colors);\n    ini.WriteBool(_T(\"config\"), _T(\"specify_each_item_color\"), m_main_wnd_data.specify_each_item_color);\n    ini.WriteInt(L\"config\", L\"hide_main_window\", m_cfg_data.m_hide_main_window);\n    ini.WriteString(L\"connection\", L\"connection_name\", CCommon::StrToUnicode(m_cfg_data.m_connection_name.c_str()));\n    ini.WriteString(_T(\"config\"), _T(\"skin_selected\"), m_cfg_data.m_skin_name.c_str());\n    ini.WriteInt(L\"config\", L\"notify_icon_selected\", m_cfg_data.m_notify_icon_selected);\n    ini.WriteBool(L\"config\", L\"notify_icon_auto_adapt\", m_cfg_data.m_notify_icon_auto_adapt);\n\n    ini.SaveFontData(L\"config\", m_main_wnd_data.font);\n\n    ini.WriteBool(L\"config\", L\"swap_up_down\", m_main_wnd_data.swap_up_down);\n    ini.WriteBool(L\"config\", L\"hide_main_wnd_when_fullscreen\", m_main_wnd_data.hide_main_wnd_when_fullscreen);\n\n    ini.WriteString(_T(\"config\"), _T(\"up_string\"), m_main_wnd_data.disp_str.Get(TDI_UP));\n    ini.WriteString(_T(\"config\"), _T(\"down_string\"), m_main_wnd_data.disp_str.Get(TDI_DOWN));\n    ini.WriteString(_T(\"config\"), _T(\"total_speed_string\"), m_main_wnd_data.disp_str.Get(TDI_TOTAL_SPEED));\n    ini.WriteString(_T(\"config\"), _T(\"cpu_string\"), m_main_wnd_data.disp_str.Get(TDI_CPU));\n    ini.WriteString(_T(\"config\"), _T(\"memory_string\"), m_main_wnd_data.disp_str.Get(TDI_MEMORY));\n    ini.WriteString(_T(\"config\"), _T(\"gpu_string\"), m_main_wnd_data.disp_str.Get(TDI_GPU_USAGE));\n    ini.WriteString(_T(\"config\"), _T(\"cpu_temp_string\"), m_main_wnd_data.disp_str.Get(TDI_CPU_TEMP));\n    ini.WriteString(_T(\"config\"), _T(\"cpu_freq_string\"), m_main_wnd_data.disp_str.Get(TDI_CPU_FREQ));\n    ini.WriteString(_T(\"config\"), _T(\"gpu_temp_string\"), m_main_wnd_data.disp_str.Get(TDI_GPU_TEMP));\n    ini.WriteString(_T(\"config\"), _T(\"hdd_temp_string\"), m_main_wnd_data.disp_str.Get(TDI_HDD_TEMP));\n    ini.WriteString(_T(\"config\"), _T(\"main_board_temp_string\"), m_main_wnd_data.disp_str.Get(TDI_MAIN_BOARD_TEMP));\n    ini.WriteString(_T(\"config\"), _T(\"hdd_string\"), m_main_wnd_data.disp_str.Get(TDI_HDD_USAGE));\n    ini.SavePluginDisplayStr(true);\n\n    ini.WriteBool(L\"config\", L\"speed_short_mode\", m_main_wnd_data.speed_short_mode);\n    ini.WriteBool(L\"config\", L\"separate_value_unit_with_space\", m_main_wnd_data.separate_value_unit_with_space);\n    ini.WriteBool(L\"config\", L\"show_tool_tip\", m_main_wnd_data.show_tool_tip);\n    ini.WriteInt(L\"config\", L\"memory_display\", static_cast<int>(m_main_wnd_data.memory_display));\n    ini.WriteBool(L\"config\", L\"unit_byte\", m_main_wnd_data.unit_byte);\n    ini.WriteInt(L\"config\", L\"speed_unit\", static_cast<int>(m_main_wnd_data.speed_unit));\n    ini.WriteBool(L\"config\", L\"hide_unit\", m_main_wnd_data.hide_unit);\n    ini.WriteBool(L\"config\", L\"hide_percent\", m_main_wnd_data.hide_percent);\n    ini.WriteInt(L\"config\", L\"double_click_action\", static_cast<int>(m_main_wnd_data.double_click_action));\n    ini.WriteString(L\"config\", L\"double_click_exe\", m_main_wnd_data.double_click_exe);\n\n    ini.WriteInt(L\"config\", L\"alow_out_of_border\", m_main_wnd_data.m_alow_out_of_border);\n\n    ini.WriteBool(L\"notify_tip\", L\"traffic_tip_enable\", m_general_data.traffic_tip_enable);\n    ini.WriteInt(L\"notify_tip\", L\"traffic_tip_value\", m_general_data.traffic_tip_value);\n    ini.WriteInt(L\"notify_tip\", L\"traffic_tip_unit\", m_general_data.traffic_tip_unit);\n    ini.WriteBool(L\"notify_tip\", L\"memory_usage_tip_enable\", m_general_data.memory_usage_tip.enable);\n    ini.WriteInt(L\"notify_tip\", L\"memory_tip_value\", m_general_data.memory_usage_tip.tip_value);\n    ini.WriteBool(L\"notify_tip\", L\"cpu_temperature_tip_enable\", m_general_data.cpu_temp_tip.enable);\n    ini.WriteInt(L\"notify_tip\", L\"cpu_temperature_tip_value\", m_general_data.cpu_temp_tip.tip_value);\n    ini.WriteBool(L\"notify_tip\", L\"gpu_temperature_tip_enable\", m_general_data.gpu_temp_tip.enable);\n    ini.WriteInt(L\"notify_tip\", L\"gpu_temperature_tip_value\", m_general_data.gpu_temp_tip.tip_value);\n    ini.WriteBool(L\"notify_tip\", L\"hdd_temperature_tip_enable\", m_general_data.hdd_temp_tip.enable);\n    ini.WriteInt(L\"notify_tip\", L\"hdd_temperature_tip_value\", m_general_data.hdd_temp_tip.tip_value);\n    ini.WriteBool(L\"notify_tip\", L\"mainboard_temperature_tip_enable\", m_general_data.mainboard_temp_tip.enable);\n    ini.WriteInt(L\"notify_tip\", L\"mainboard_temperature_tip_value\", m_general_data.mainboard_temp_tip.tip_value);\n\n    //任务栏窗口设置\n    ini.WriteInt(L\"task_bar\", L\"task_bar_back_color\", m_taskbar_data.back_color);\n    ini.WriteInt(L\"task_bar\", L\"transparent_color\", m_taskbar_data.transparent_color);\n    ini.WriteInt(L\"task_bar\", L\"status_bar_color\", m_taskbar_data.status_bar_color);\n    ini.SaveTaskbarWndColors(L\"task_bar\", L\"task_bar_text_color\", m_taskbar_data.text_colors);\n    ini.WriteBool(L\"task_bar\", L\"specify_each_item_color\", m_taskbar_data.specify_each_item_color);\n    //ini.WriteBool(L\"task_bar\", L\"task_bar_show_cpu_memory\", m_cfg_data.m_tbar_show_cpu_memory);\n    ini.WriteInt(L\"task_bar\", L\"tbar_display_item\", m_taskbar_data.m_tbar_display_item);\n    ini.SaveFontData(L\"task_bar\", m_taskbar_data.font);\n    //ini.WriteBool(L\"task_bar\", L\"task_bar_swap_up_down\", m_taskbar_data.swap_up_down);\n\n    ini.WriteString(_T(\"task_bar\"), _T(\"up_string\"), m_taskbar_data.disp_str.Get(TDI_UP));\n    ini.WriteString(_T(\"task_bar\"), _T(\"down_string\"), m_taskbar_data.disp_str.Get(TDI_DOWN));\n    ini.WriteString(_T(\"task_bar\"), _T(\"total_speed_string\"), m_taskbar_data.disp_str.Get(TDI_TOTAL_SPEED));\n    ini.WriteString(_T(\"task_bar\"), _T(\"cpu_string\"), m_taskbar_data.disp_str.Get(TDI_CPU));\n    ini.WriteString(_T(\"task_bar\"), _T(\"memory_string\"), m_taskbar_data.disp_str.Get(TDI_MEMORY));\n    ini.WriteString(_T(\"task_bar\"), _T(\"gpu_string\"), m_taskbar_data.disp_str.Get(TDI_GPU_USAGE));\n    ini.WriteString(_T(\"task_bar\"), _T(\"cpu_temp_string\"), m_taskbar_data.disp_str.Get(TDI_CPU_TEMP));\n    ini.WriteString(_T(\"task_bar\"), _T(\"cpu_freq_string\"), m_taskbar_data.disp_str.Get(TDI_CPU_FREQ));\n    ini.WriteString(_T(\"task_bar\"), _T(\"gpu_temp_string\"), m_taskbar_data.disp_str.Get(TDI_GPU_TEMP));\n    ini.WriteString(_T(\"task_bar\"), _T(\"hdd_temp_string\"), m_taskbar_data.disp_str.Get(TDI_HDD_TEMP));\n    ini.WriteString(_T(\"task_bar\"), _T(\"main_board_temp_string\"), m_taskbar_data.disp_str.Get(TDI_MAIN_BOARD_TEMP));\n    ini.WriteString(_T(\"task_bar\"), _T(\"hdd_string\"), m_taskbar_data.disp_str.Get(TDI_HDD_USAGE));\n    ini.SavePluginDisplayStr(false);\n\n    ini.WriteBool(L\"task_bar\", L\"task_bar_wnd_on_left\", m_taskbar_data.tbar_wnd_on_left);\n    ini.WriteBool(L\"task_bar\", L\"task_bar_wnd_snap\", m_taskbar_data.tbar_wnd_snap);\n    ini.WriteBool(L\"task_bar\", L\"task_bar_speed_short_mode\", m_taskbar_data.speed_short_mode);\n    ini.WriteBool(L\"task_bar\", L\"unit_byte\", m_taskbar_data.unit_byte);\n    ini.WriteInt(L\"task_bar\", L\"task_bar_speed_unit\", static_cast<int>(m_taskbar_data.speed_unit));\n    ini.WriteBool(L\"task_bar\", L\"task_bar_hide_unit\", m_taskbar_data.hide_unit);\n    ini.WriteBool(L\"task_bar\", L\"task_bar_hide_percent\", m_taskbar_data.hide_percent);\n    ini.WriteBool(L\"task_bar\", L\"value_right_align\", m_taskbar_data.value_right_align);\n    ini.WriteBool(L\"task_bar\", L\"horizontal_arrange\", m_taskbar_data.horizontal_arrange);\n    ini.WriteBool(L\"task_bar\", L\"show_status_bar\", m_taskbar_data.show_status_bar);\n    ini.WriteBool(L\"task_bar\", L\"separate_value_unit_with_space\", m_taskbar_data.separate_value_unit_with_space);\n    ini.WriteBool(L\"task_bar\", L\"show_tool_tip\", m_taskbar_data.show_tool_tip);\n    ini.WriteInt(L\"task_bar\", L\"digits_number\", m_taskbar_data.digits_number);\n    ini.WriteInt(L\"task_bar\", L\"memory_display\", static_cast<int>(m_taskbar_data.memory_display));\n    ini.WriteInt(L\"task_bar\", L\"double_click_action\", static_cast<int>(m_taskbar_data.double_click_action));\n    ini.WriteString(L\"task_bar\", L\"double_click_exe\", m_taskbar_data.double_click_exe);\n    ini.WriteBool(L\"task_bar\", L\"cm_graph_type\", m_taskbar_data.cm_graph_type);\n    ini.WriteBool(L\"task_bar\", L\"show_graph_dashed_box\", m_taskbar_data.show_graph_dashed_box);\n    ini.WriteInt(L\"task_bar\", L\"item_space\", m_taskbar_data.item_space);\n\n    ini.WriteBool(L\"task_bar\", L\"auto_adapt_light_theme\", m_taskbar_data.auto_adapt_light_theme);\n    ini.WriteInt(L\"task_bar\", L\"dark_default_style\", m_taskbar_data.dark_default_style);\n    ini.WriteInt(L\"task_bar\", L\"light_default_style\", m_taskbar_data.light_default_style);\n    ini.WriteBool(L\"task_bar\", L\"auto_set_background_color\", m_taskbar_data.auto_set_background_color);\n\n    ini.WriteString(L\"task_bar\", L\"item_order\", m_taskbar_data.item_order.ToString());\n    ini.WriteString(L\"task_bar\", L\"plugin_display_item\", m_taskbar_data.plugin_display_item.ToString());\n    ini.WriteBool(L\"task_bar\", L\"auto_save_taskbar_color_settings_to_preset\", m_taskbar_data.auto_save_taskbar_color_settings_to_preset);\n\n    ini.WriteBool(L\"task_bar\", L\"show_netspeed_figure\", m_taskbar_data.show_netspeed_figure);\n    ini.WriteInt(L\"task_bar\", L\"netspeed_figure_max_value\", m_taskbar_data.netspeed_figure_max_value);\n    ini.WriteInt(L\"task_bar\", L\"netspeed_figure_max_value_unit\", m_taskbar_data.netspeed_figure_max_value_unit);\n\n    //其他设置\n    //ini.WriteBool(L\"connection_details\", L\"show_internet_ip\", m_cfg_data.m_show_internet_ip);\n    ini.WriteBool(L\"histroy_traffic\", L\"use_log_scale\", m_cfg_data.m_use_log_scale);\n    ini.WriteBool(L\"histroy_traffic\", L\"sunday_first\", m_cfg_data.m_sunday_first);\n    ini.WriteInt(L\"histroy_traffic\", L\"view_type\", static_cast<int>(m_cfg_data.m_view_type));\n\n    ini.WriteBool(_T(\"other\"), _T(\"no_multistart_warning\"), m_no_multistart_warning);\n    ini.WriteBool(_T(\"other\"), _T(\"exit_when_start_by_restart_manager\"), m_exit_when_start_by_restart_manager);\n    ini.WriteBool(_T(\"other\"), _T(\"debug_log\"), m_debug_log);\n    ini.WriteInt(_T(\"other\"), _T(\"notify_interval\"), m_notify_interval);\n    ini.WriteBool(_T(\"other\"), _T(\"taksbar_transparent_color_enable\"), m_taksbar_transparent_color_enable);\n    ini.WriteBool(_T(\"other\"), _T(\"last_light_mode\"), m_last_light_mode);\n    ini.WriteBool(_T(\"other\"), _T(\"show_mouse_panetrate_tip\"), m_show_mouse_panetrate_tip);\n    ini.WriteBool(_T(\"other\"), _T(\"show_dot_net_notinstalled_tip\"), m_show_dot_net_notinstalled_tip);\n\n    ini.WriteString(L\"config\", L\"plugin_disabled\", m_cfg_data.plugin_disabled.ToString());\n\n    ini.WriteInt(L\"task_bar\", L\"taskbar_left_space_win11\", m_cfg_data.taskbar_left_space_win11);\n\n    ini.WriteString(L\"app\", L\"version\", VERSION);\n\n    //检查是否保存成功\n    if (!ini.Save())\n    {\n        if (m_cannot_save_config_warning)\n        {\n            CString info;\n            info.LoadString(IDS_CONNOT_SAVE_CONFIG_WARNING);\n            info.Replace(_T(\"<%file_path%>\"), m_config_path.c_str());\n            AfxMessageBox(info, MB_ICONWARNING);\n        }\n        m_cannot_save_config_warning = false;\n        return;\n    }\n}\n\nvoid CTrafficMonitorApp::LoadPluginDisabledSettings()\n{\n    CIniHelper ini{ m_config_path };\n    m_cfg_data.plugin_disabled.FromString(ini.GetString(L\"config\", L\"plugin_disabled\", L\"\"));\n}\n\nvoid CTrafficMonitorApp::LoadGlobalConfig()\n{\n    bool portable_mode_default{ false };\n    wstring global_cfg_path{ m_module_dir + L\"global_cfg.ini\" };\n    if (!CCommon::FileExist(global_cfg_path.c_str()))       //如果global_cfg.ini不存在，则根据AppData/Roaming/TrafficMonitor目录下是否存在config.ini来判断配置文件的保存位置\n    {\n        portable_mode_default = !CCommon::FileExist((m_appdata_dir + L\"config.ini\").c_str());\n    }\n\n    CIniHelper ini{ global_cfg_path };\n    m_general_data.portable_mode = ini.GetBool(L\"config\", L\"portable_mode\", portable_mode_default);\n\n    //执行一次保存操作，以检查当前目录是否有写入权限\n    m_module_dir_writable = ini.Save();\n\n    if (m_module_dir.find(CCommon::GetTemplateDir()) != wstring::npos)      //如果当前路径是在Temp目录下，则强制将数据保存到Appdata\n    {\n        m_module_dir_writable = false;\n    }\n\n    if (!m_module_dir_writable)              //如果当前目录没有写入权限，则设置配置保存到AppData目录\n    {\n        m_general_data.portable_mode = false;\n    }\n}\n\nvoid CTrafficMonitorApp::SaveGlobalConfig()\n{\n    CIniHelper ini{ m_module_dir + L\"global_cfg.ini\" };\n    ini.WriteBool(L\"config\", L\"portable_mode\", m_general_data.portable_mode);\n\n    //检查是否保存成功\n    if (!ini.Save())\n    {\n        //if (m_cannot_save_global_config_warning)\n        //{\n        //    CString info;\n        //    info.LoadString(IDS_CONNOT_SAVE_CONFIG_WARNING);\n        //    info.Replace(_T(\"<%file_path%>\"), m_module_dir.c_str());\n        //    AfxMessageBox(info, MB_ICONWARNING);\n        //}\n        //m_cannot_save_global_config_warning = false;\n        //return;\n    }\n}\n\nint CTrafficMonitorApp::DPI(int pixel)\n{\n    return m_dpi * pixel / 96;\n}\n\nvoid CTrafficMonitorApp::DPI(CRect& rect)\n{\n    rect.left = DPI(rect.left);\n    rect.right = DPI(rect.right);\n    rect.top = DPI(rect.top);\n    rect.bottom = DPI(rect.bottom);\n}\n\nvoid CTrafficMonitorApp::DPIFromWindow(CWnd* pWnd)\n{\n    CWindowDC dc(pWnd);\n    HDC hDC = dc.GetSafeHdc();\n    m_dpi = GetDeviceCaps(hDC, LOGPIXELSY);\n}\n\nvoid CTrafficMonitorApp::CheckUpdate(bool message)\n{\n    if (m_checking_update)      //如果还在检查更新，则直接返回\n        return;\n    CFlagLocker update_locker(m_checking_update);\n    CWaitCursor wait_cursor;\n\n    wstring version;        //程序版本\n    wstring link;           //下载链接\n    wstring contents_zh_cn; //更新内容（简体中文）\n    wstring contents_en;    //更新内容（English）\n    wstring contents_zh_tw; //更新内容（繁体中文）\n    CUpdateHelper update_helper;\n    update_helper.SetUpdateSource(static_cast<CUpdateHelper::UpdateSource>(m_general_data.update_source));\n    if (!update_helper.CheckForUpdate())\n    {\n        if (message)\n            AfxMessageBox(CCommon::LoadText(IDS_CHECK_UPDATE_FAILD), MB_OK | MB_ICONWARNING);\n        return;\n    }\n    version = update_helper.GetVersion();\n#ifdef _M_X64\n    link = update_helper.GetLink64();\n#else\n    link = update_helper.GetLink();\n#endif\n    contents_zh_cn = update_helper.GetContentsZhCn();\n    contents_en = update_helper.GetContentsEn();\n    contents_zh_tw = update_helper.GetContentsZhTw();\n    if (version.empty() || link.empty())\n    {\n        if (message)\n        {\n            CString info = CCommon::LoadText(IDS_CHECK_UPDATE_ERROR);\n            info += _T(\"\\r\\nrow_data=\");\n            info += std::to_wstring(update_helper.IsRowData()).c_str();\n\n            AfxMessageBox(info, MB_OK | MB_ICONWARNING);\n        }\n        return;\n    }\n    if (version > VERSION)      //如果服务器上的版本大于本地版本\n    {\n        CString info;\n        //根据语言设置选择对应语言版本的更新内容\n        int language_code = _ttoi(CCommon::LoadText(IDS_LANGUAGE_CODE));\n        wstring contents_lan;\n        switch (language_code)\n        {\n        case 2: contents_lan = contents_zh_cn; break;\n        case 3: contents_lan = contents_zh_tw; break;\n        default: contents_lan = contents_en; break;\n        }\n\n        if (contents_lan.empty())\n            info.Format(CCommon::LoadText(IDS_UPDATE_AVLIABLE), version.c_str());\n        else\n            info.Format(CCommon::LoadText(IDS_UPDATE_AVLIABLE2), version.c_str(), contents_lan.c_str());\n\n        if (AfxMessageBox(info, MB_YESNO | MB_ICONQUESTION) == IDYES)\n        {\n            ShellExecute(NULL, _T(\"open\"), link.c_str(), NULL, NULL, SW_SHOW);      //转到下载链接\n        }\n    }\n    else\n    {\n        if (message)\n            AfxMessageBox(CCommon::LoadText(IDS_ALREADY_UPDATED), MB_OK | MB_ICONINFORMATION);\n    }\n}\n\nvoid CTrafficMonitorApp::CheckUpdateInThread(bool message)\n{\n    AfxBeginThread(CheckUpdateThreadFunc, (LPVOID)message);\n}\n\nUINT CTrafficMonitorApp::CheckUpdateThreadFunc(LPVOID lpParam)\n{\n    CCommon::SetThreadLanguage(theApp.m_general_data.language);     //设置线程语言\n    theApp.CheckUpdate(lpParam);        //检查更新\n    return 0;\n}\n\nUINT CTrafficMonitorApp::InitOpenHardwareMonitorLibThreadFunc(LPVOID lpParam)\n{\n#ifndef WITHOUT_TEMPERATURE\n    CSingleLock sync(&theApp.m_minitor_lib_critical, TRUE);\n    theApp.m_pMonitor = OpenHardwareMonitorApi::CreateInstance();\n    if (theApp.m_pMonitor == nullptr)\n    {\n        AfxMessageBox(OpenHardwareMonitorApi::GetErrorMessage().c_str(), MB_ICONERROR | MB_OK);\n    }\n    //设置硬件监控的启用状态\n    theApp.UpdateOpenHardwareMonitorEnableState();\n#endif\n    return 0;\n}\n\nbool  CTrafficMonitorApp::SetAutoRun(bool auto_run)\n{\n    //不含温度监控的版本使用添加注册表项的方式实现开机自启动\n#ifdef WITHOUT_TEMPERATURE\n    return SetAutoRunByRegistry(auto_run);\n#else\n    //包含温度监控的版本使用任务计划的方式实现开机自启动\n    return SetAutoRunByTaskScheduler(auto_run);\n#endif\n}\n\nbool CTrafficMonitorApp::GetAutoRun(wstring* auto_run_path)\n{\n    if (auto_run_path != nullptr)\n        auto_run_path->clear();\n    //不含温度监控的版本使用添加注册表项的方式实现开机自启动\n#ifdef WITHOUT_TEMPERATURE\n    CRegKey key;\n    if (key.Open(HKEY_CURRENT_USER, _T(\"Software\\\\Microsoft\\\\Windows\\\\CurrentVersion\\\\Run\")) != ERROR_SUCCESS)\n    {\n        //打开注册表“Software\\\\Microsoft\\\\Windows\\\\CurrentVersion\\\\Run”失败，则返回false\n        return false;\n    }\n    wchar_t buff[256];\n    ULONG size{ 256 };\n    if (key.QueryStringValue(APP_NAME, buff, &size) == ERROR_SUCCESS)       //如果找到了“TrafficMonitor”键\n    {\n        if (auto_run_path != nullptr)\n        {\n            //保存路径\n            *auto_run_path = buff;\n            //去掉前后的引号\n            if (auto_run_path->front() == L'\\\"')\n                *auto_run_path = auto_run_path->substr(1);\n            if (auto_run_path->back() = L'\\\"')\n                auto_run_path->pop_back();\n        }\n        return (m_module_path_reg == buff); //如果“TrafficMonitor”的值是当前程序的路径，就返回true，否则返回false\n    }\n    else\n    {\n        return false;       //没有找到“TrafficMonitor”键，返回false\n    }\n#else\n    //包含温度监控的版本使用任务计划的方式实现开机自启动\n    return is_auto_start_task_active_for_this_user(auto_run_path);\n#endif\n}\n\nbool CTrafficMonitorApp::SetAutoRunByRegistry(bool auto_run)\n{\n    CRegKey key;\n    //打开注册表项\n    if (key.Open(HKEY_CURRENT_USER, _T(\"Software\\\\Microsoft\\\\Windows\\\\CurrentVersion\\\\Run\")) != ERROR_SUCCESS)\n    {\n        AfxMessageBox(CCommon::LoadText(IDS_AUTORUN_FAILED_NO_KEY), MB_OK | MB_ICONWARNING);\n        return false;\n    }\n    if (auto_run)       //写入注册表项\n    {\n        //通过注册表设置开机自启动项时删除计划任务中的自启动项\n        SetAutoRunByTaskScheduler(false);\n\n        if (key.SetStringValue(APP_NAME, m_module_path_reg.c_str()) != ERROR_SUCCESS)\n        {\n            AfxMessageBox(CCommon::LoadText(IDS_AUTORUN_FAILED_NO_ACCESS), MB_OK | MB_ICONWARNING);\n            return false;\n        }\n    }\n    else        //删除注册表项\n    {\n        //删除前先检查注册表项是否存在，如果不存在，则直接返回\n        wchar_t buff[256];\n        ULONG size{ 256 };\n        if (key.QueryStringValue(APP_NAME, buff, &size) != ERROR_SUCCESS)\n            return false;\n        if (key.DeleteValue(APP_NAME) != ERROR_SUCCESS)\n        {\n            AfxMessageBox(CCommon::LoadText(IDS_AUTORUN_DELETE_FAILED), MB_OK | MB_ICONWARNING);\n            return false;\n        }\n    }\n    return true;\n}\n\nbool CTrafficMonitorApp::SetAutoRunByTaskScheduler(bool auto_run)\n{\n    bool succeed = delete_auto_start_task_for_this_user();     //先删除开机自启动\n    if (auto_run)\n    {\n        //通过计划任务设置开机自启动项时删除注册表中的自启动项\n        SetAutoRunByRegistry(false);\n\n        succeed = create_auto_start_task_for_this_user(true);\n    }\n    return succeed;\n}\n\nCString CTrafficMonitorApp::GetSystemInfoString()\n{\n    CString info;\n    info += _T(\"System Info:\\r\\n\");\n\n    CString strTmp;\n    strTmp.Format(_T(\"Windows Version: %d.%d build %d\\r\\n\"), m_win_version.GetMajorVersion(),\n        m_win_version.GetMinorVersion(), m_win_version.GetBuildNumber());\n    info += strTmp;\n\n    strTmp.Format(_T(\"DPI: %d\"), m_dpi);\n    info += strTmp;\n    info += _T(\"\\r\\n\");\n\n    info += _T(\"Version: \");\n    info += VERSION;\n    info += _T(\" \");\n#ifdef _M_X64\n    info += _T(\"x64\");\n#else\n    info += _T(\"x86\");\n#endif\n\n#ifdef WITHOUT_TEMPERATURE\n    info += CCommon::LoadText(_T(\" (\"), IDS_WITHOUT_TEMPERATURE, _T(\")\"));\n#endif\n\n    info += _T(\"\\r\\nLast compiled date: \");\n    info += CCommon::GetLastCompileTime();\n\n    return info;\n}\n\n\nvoid CTrafficMonitorApp::InitMenuResourse()\n{\n    //载入菜单\n    m_main_menu.LoadMenu(IDR_MENU1);\n    m_taskbar_menu.LoadMenu(IDR_TASK_BAR_MENU);\n\n    //为菜单项添加图标\n    //主窗口右键菜单\n    CMenuIcon::AddIconToMenuItem(m_main_menu.GetSubMenu(0)->GetSafeHmenu(), 0, TRUE, GetMenuIcon(IDI_CONNECTION));\n    CMenuIcon::AddIconToMenuItem(m_main_menu.GetSubMenu(0)->GetSafeHmenu(), 11, TRUE, GetMenuIcon(IDI_FUNCTION));\n    CMenuIcon::AddIconToMenuItem(m_main_menu.GetSafeHmenu(), ID_NETWORK_INFO, FALSE, GetMenuIcon(IDI_INFO));\n    if (!m_win_version.IsWindows11OrLater())\n        CMenuIcon::AddIconToMenuItem(m_main_menu.GetSafeHmenu(), ID_ALWAYS_ON_TOP, FALSE, GetMenuIcon(IDI_PIN));\n    if (!m_win_version.IsWindows11OrLater())\n        CMenuIcon::AddIconToMenuItem(m_main_menu.GetSafeHmenu(), ID_MOUSE_PENETRATE, FALSE, GetMenuIcon(IDI_MOUSE));\n    if (!m_win_version.IsWindows11OrLater())\n        CMenuIcon::AddIconToMenuItem(m_main_menu.GetSafeHmenu(), ID_LOCK_WINDOW_POS, FALSE, GetMenuIcon(IDI_LOCK));\n    //if (!m_win_version.IsWindows11OrLater())\n    //    CMenuIcon::AddIconToMenuItem(m_main_menu.GetSafeHmenu(), ID_SHOW_NOTIFY_ICON, FALSE, GetMenuIcon(IDI_NOTIFY));\n    if (!m_win_version.IsWindows11OrLater())\n        CMenuIcon::AddIconToMenuItem(m_main_menu.GetSafeHmenu(), ID_SHOW_CPU_MEMORY, FALSE, GetMenuIcon(IDI_MORE));\n    if (!m_win_version.IsWindows11OrLater())\n        CMenuIcon::AddIconToMenuItem(m_main_menu.GetSafeHmenu(), ID_SHOW_TASK_BAR_WND, FALSE, GetMenuIcon(IDI_TASKBAR_WINDOW));\n    if (!m_win_version.IsWindows11OrLater())\n        CMenuIcon::AddIconToMenuItem(m_main_menu.GetSafeHmenu(), ID_SHOW_MAIN_WND, FALSE, GetMenuIcon(IDI_MAIN_WINDOW));\n    CMenuIcon::AddIconToMenuItem(m_main_menu.GetSafeHmenu(), ID_CHANGE_SKIN, FALSE, GetMenuIcon(IDI_SKIN));\n    CMenuIcon::AddIconToMenuItem(m_main_menu.GetSafeHmenu(), ID_CHANGE_NOTIFY_ICON, FALSE, GetMenuIcon(IDI_NOTIFY));\n    CMenuIcon::AddIconToMenuItem(m_main_menu.GetSafeHmenu(), ID_TRAFFIC_HISTORY, FALSE, GetMenuIcon(IDI_STATISTICS));\n    CMenuIcon::AddIconToMenuItem(m_main_menu.GetSafeHmenu(), ID_PLUGIN_MANAGE, FALSE, GetMenuIcon(IDI_PLUGINS));\n    CMenuIcon::AddIconToMenuItem(m_main_menu.GetSafeHmenu(), ID_OPTIONS, FALSE, GetMenuIcon(IDI_SETTINGS));\n    CMenuIcon::AddIconToMenuItem(m_main_menu.GetSubMenu(0)->GetSafeHmenu(), 14, TRUE, GetMenuIcon(IDI_HELP));\n    CMenuIcon::AddIconToMenuItem(m_main_menu.GetSafeHmenu(), ID_HELP, FALSE, GetMenuIcon(IDI_HELP));\n    CMenuIcon::AddIconToMenuItem(m_main_menu.GetSafeHmenu(), ID_APP_ABOUT, FALSE, GetMenuIcon(IDR_MAINFRAME));\n    CMenuIcon::AddIconToMenuItem(m_main_menu.GetSafeHmenu(), ID_APP_EXIT, FALSE, GetMenuIcon(IDI_EXIT));\n\n    //任务栏窗口右键菜单\n    CMenuIcon::AddIconToMenuItem(m_taskbar_menu.GetSubMenu(0)->GetSafeHmenu(), 0, TRUE, GetMenuIcon(IDI_CONNECTION));\n    CMenuIcon::AddIconToMenuItem(m_taskbar_menu.GetSafeHmenu(), ID_NETWORK_INFO, FALSE, GetMenuIcon(IDI_INFO));\n    CMenuIcon::AddIconToMenuItem(m_taskbar_menu.GetSafeHmenu(), ID_TRAFFIC_HISTORY, FALSE, GetMenuIcon(IDI_STATISTICS));\n    CMenuIcon::AddIconToMenuItem(m_taskbar_menu.GetSafeHmenu(), ID_DISPLAY_SETTINGS, FALSE, GetMenuIcon(IDI_ITEM));\n    //if (!m_win_version.IsWindows11OrLater())\n    //    CMenuIcon::AddIconToMenuItem(m_taskbar_menu.GetSafeHmenu(), ID_SHOW_NOTIFY_ICON, FALSE, GetMenuIcon(IDI_NOTIFY));\n    if (!m_win_version.IsWindows11OrLater())\n        CMenuIcon::AddIconToMenuItem(m_taskbar_menu.GetSafeHmenu(), ID_SHOW_MAIN_WND, FALSE, GetMenuIcon(IDI_MAIN_WINDOW));\n    CMenuIcon::AddIconToMenuItem(m_taskbar_menu.GetSafeHmenu(), ID_SHOW_TASK_BAR_WND, FALSE, GetMenuIcon(IDI_CLOSE));\n    CMenuIcon::AddIconToMenuItem(m_taskbar_menu.GetSafeHmenu(), ID_OPEN_TASK_MANAGER, FALSE, GetMenuIcon(IDI_TASK_MANAGER));\n    CMenuIcon::AddIconToMenuItem(m_taskbar_menu.GetSafeHmenu(), ID_OPTIONS2, FALSE, GetMenuIcon(IDI_SETTINGS));\n    CMenuIcon::AddIconToMenuItem(m_taskbar_menu.GetSubMenu(0)->GetSafeHmenu(), 12, TRUE, GetMenuIcon(IDI_HELP));\n    CMenuIcon::AddIconToMenuItem(m_taskbar_menu.GetSafeHmenu(), ID_HELP, FALSE, GetMenuIcon(IDI_HELP));\n    CMenuIcon::AddIconToMenuItem(m_taskbar_menu.GetSafeHmenu(), ID_APP_ABOUT, FALSE, GetMenuIcon(IDR_MAINFRAME));\n    CMenuIcon::AddIconToMenuItem(m_taskbar_menu.GetSafeHmenu(), ID_APP_EXIT, FALSE, GetMenuIcon(IDI_EXIT));\n\n#ifdef _DEBUG\n    m_main_menu.GetSubMenu(0)->AppendMenu(MF_BYCOMMAND, ID_CMD_TEST, _T(\"Test Command\"));\n#endif\n}\n\nHICON CTrafficMonitorApp::GetMenuIcon(UINT id)\n{\n    auto iter = m_menu_icons.find(id);\n    if (iter != m_menu_icons.end())\n    {\n        return iter->second;\n    }\n    else\n    {\n        HICON hIcon = CCommon::LoadIconResource(id, DPI(16));\n        m_menu_icons[id] = hIcon;\n        return hIcon;\n    }\n}\n\nvoid CTrafficMonitorApp::AutoSelectNotifyIcon()\n{\n    if (m_win_version.GetMajorVersion() >= 10)\n    {\n        bool light_mode = CWindowsSettingHelper::IsWindows10LightTheme();\n        if (light_mode)     //浅色模式下，如果图标是白色，则改成黑色\n        {\n            if (m_cfg_data.m_notify_icon_selected == 0)\n                m_cfg_data.m_notify_icon_selected = 4;\n            if (m_cfg_data.m_notify_icon_selected == 1)\n                m_cfg_data.m_notify_icon_selected = 5;\n        }\n        else     //深色模式下，如果图标是黑色，则改成白色\n        {\n            if (m_cfg_data.m_notify_icon_selected == 4)\n                m_cfg_data.m_notify_icon_selected = 0;\n            if (m_cfg_data.m_notify_icon_selected == 5)\n                m_cfg_data.m_notify_icon_selected = 1;\n        }\n    }\n}\n\n// 唯一的一个 CTrafficMonitorApp 对象\n\nCTrafficMonitorApp theApp;\n\n\n// CTrafficMonitorApp 初始化\n\nBOOL CTrafficMonitorApp::InitInstance()\n{\n    //替换掉对话框程序的默认类名\n    WNDCLASS wc;\n    ::GetClassInfo(AfxGetInstanceHandle(), _T(\"#32770\"), &wc);       //MFC默认的所有对话框的类名为#32770\n    wc.lpszClassName = APP_CLASS_NAME;      //将对话框的类名修改为新类名\n    AfxRegisterClass(&wc);\n\n    //设置配置文件的路径\n    wchar_t path[MAX_PATH];\n    GetModuleFileNameW(NULL, path, MAX_PATH);\n    m_module_path = path;\n    if (m_module_path.find(L' ') != wstring::npos)\n    {\n        //如果路径中有空格，则在程序路径前后添加双引号\n        m_module_path_reg = L'\\\"';\n        m_module_path_reg += m_module_path;\n        m_module_path_reg += L'\\\"';\n    }\n    else\n    {\n        m_module_path_reg = m_module_path;\n    }\n    m_module_dir = CCommon::GetModuleDir();\n    m_system_dir = CCommon::GetSystemDir();\n    m_appdata_dir = CCommon::GetAppDataConfigDir();\n\n    LoadGlobalConfig();\n\n#ifdef _DEBUG\n    m_config_dir = L\".\\\\\";\n    m_skin_path = L\".\\\\skins\";\n#else\n    if (m_general_data.portable_mode)\n        m_config_dir = m_module_dir;\n    else\n        m_config_dir = m_appdata_dir;\n    m_skin_path = m_module_dir + L\"skins\";\n#endif\n    //AppData里面的程序配置文件路径\n    m_config_path = m_config_dir + L\"config.ini\";\n    m_history_traffic_path = m_config_dir + L\"history_traffic.dat\";\n    m_log_path = m_config_dir + L\"error.log\";\n\n    //#ifndef _DEBUG\n    //  //原来的、程序所在目录下的配置文件的路径\n    //  wstring config_path_old = m_module_dir + L\"config.ini\";\n    //  wstring history_traffic_path_old = m_module_dir + L\"history_traffic.dat\";\n    //  wstring log_path_old = m_module_dir + L\"error.log\";\n    //  //如果程序所在目录下含有配置文件，则将其移动到AppData对应的目录下面\n    //  CCommon::MoveAFile(config_path_old.c_str(), m_config_path.c_str());\n    //  CCommon::MoveAFile(history_traffic_path_old.c_str(), m_history_traffic_path.c_str());\n    //  CCommon::MoveAFile(log_path_old.c_str(), m_log_path.c_str());\n    //#endif // !_DEBUG\n\n    bool is_windows10_fall_creator = m_win_version.IsWindows10FallCreatorOrLater();\n\n    //载入插件\n    LoadPluginDisabledSettings();\n    m_plugins.LoadPlugins();\n\n    //向插件传递配置文件的路径\n    wstring plugin_dir;\n#ifdef _DEBUG\n    plugin_dir = m_module_dir;\n#else\n    plugin_dir = m_config_dir;\n#endif\n    plugin_dir += L\"plugins\\\\\";\n    m_plugins.EnumPlugin([&](ITMPlugin* plugin)\n        {\n            if (plugin->GetAPIVersion() >= 2)\n            {\n                CreateDirectory(plugin_dir.c_str(), NULL);       //如果plugins不存在，则创建它\n                plugin->OnExtenedInfo(ITMPlugin::EI_CONFIG_DIR, plugin_dir.c_str());\n            }\n        });\n\n    //从ini文件载入设置\n    LoadConfig();\n\n    //初始化界面语言\n    CCommon::SetThreadLanguage(m_general_data.language);\n\n    //wstring cmd_line{ m_lpCmdLine };\n    //bool is_restart{ cmd_line.find(L\"RestartByRestartManager\") != wstring::npos };        //如果命令行参数中含有字符串“RestartByRestartManager”则说明程序是被Windows重新启动的\n    ////bool when_start{ CCommon::WhenStart(m_no_multistart_warning_time) };\n    //if (m_exit_when_start_by_restart_manager && is_restart && is_windows10_fall_creator)      //当前Windows版本是秋季创意者更新时，如果程序被重新启动，则直接退出程序\n    //{\n    //  //AfxMessageBox(_T(\"调试信息：程序已被Windows的重启管理器重新启动。\"));\n    //  return FALSE;\n    //}\n\n    //检查是否已有实例正在运行\n    LPCTSTR mutex_name{};\n#ifdef _DEBUG\n    mutex_name = _T(\"TrafficMonitor-e8Ahk24HP6JC8hDy\");\n#else\n    mutex_name = _T(\"TrafficMonitor-1419J3XLKL1w8OZc\");\n#endif\n    HANDLE hMutex = ::CreateMutex(NULL, TRUE, mutex_name);\n    if (hMutex != NULL)\n    {\n        if (GetLastError() == ERROR_ALREADY_EXISTS)\n        {\n            //char buff[128];\n            //string cmd_line_str{ CCommon::UnicodeToStr(cmd_line.c_str()) };\n            //sprintf_s(buff, \"when_start=%d, m_no_multistart_warning=%d, cmd_line=%s\", when_start, m_no_multistart_warning, cmd_line_str.c_str());\n            //CCommon::WriteLog(buff, _T(\".\\\\start.log\"));\n            if (!m_no_multistart_warning)\n            {\n                //查找已存在TrafficMonitor进程的主窗口的句柄\n                HWND exist_handel = ::FindWindow(APP_CLASS_NAME, NULL);\n                if (exist_handel != NULL)\n                {\n                    //弹出“TrafficMonitor已在运行”对话框\n                    CAppAlreadyRuningDlg dlg(exist_handel);\n                    dlg.DoModal();\n                }\n                else\n                {\n                    AfxMessageBox(CCommon::LoadText(IDS_AN_INSTANCE_RUNNING));\n                }\n            }\n            return FALSE;\n        }\n    }\n\n    m_taskbar_default_style.LoadConfig();\n\n    //SaveConfig();\n\n    // 如果一个运行在 Windows XP 上的应用程序清单指定要\n    // 使用 ComCtl32.dll 版本 6 或更高版本来启用可视化方式，\n    //则需要 InitCommonControlsEx()。  否则，将无法创建窗口。\n    INITCOMMONCONTROLSEX InitCtrls;\n    InitCtrls.dwSize = sizeof(InitCtrls);\n    // 将它设置为包括所有要在应用程序中使用的\n    // 公共控件类。\n    InitCtrls.dwICC = ICC_WIN95_CLASSES;\n    InitCommonControlsEx(&InitCtrls);\n\n    CWinApp::InitInstance();\n\n\n    AfxEnableControlContainer();\n\n    // 创建 shell 管理器，以防对话框包含\n    // 任何 shell 树视图控件或 shell 列表视图控件。\n    CShellManager* pShellManager = new CShellManager;\n\n    // 激活“Windows Native”视觉管理器，以便在 MFC 控件中启用主题\n    CMFCVisualManager::SetDefaultManager(RUNTIME_CLASS(CMFCVisualManagerWindows));\n\n    // 标准初始化\n    // 如果未使用这些功能并希望减小\n    // 最终可执行文件的大小，则应移除下列\n    // 不需要的特定初始化例程\n    // 更改用于存储设置的注册表项\n    // TODO: 应适当修改该字符串，\n    // 例如修改为公司或组织名\n    //SetRegistryKey(_T(\"应用程序向导生成的本地应用程序\"));        //暂不使用注册表保存数据\n\n    //启动时检查更新\n#ifndef _DEBUG      //DEBUG下不在启动时检查更新\n    if (m_general_data.check_update_when_start)\n    {\n        CheckUpdateInThread(false);\n    }\n#endif // !_DEBUG\n\n#ifndef WITHOUT_TEMPERATURE\n    //检测是否安装.net framework 4.5\n    if (!CWindowsSettingHelper::IsDotNetFramework4Point5Installed())\n    {\n        if (theApp.m_show_dot_net_notinstalled_tip)\n        {\n            if (AfxMessageBox(CCommon::LoadText(IDS_DOTNET_NOT_INSTALLED_TIP), MB_OKCANCEL | MB_ICONWARNING) == IDCANCEL)       //点击“取消”不再提示\n            {\n                theApp.m_show_dot_net_notinstalled_tip = false;\n                SaveConfig();\n            }\n        }\n    }\n    else\n    {\n        //如果没有开启任何一项的硬件监控，则不初始化OpenHardwareMonitor\n        if (theApp.m_general_data.IsHardwareEnable(HI_CPU) || theApp.m_general_data.IsHardwareEnable(HI_GPU)\n            || theApp.m_general_data.IsHardwareEnable(HI_HDD) || theApp.m_general_data.IsHardwareEnable(HI_MBD))\n        {\n            //启动初始化OpenHardwareMonitor的线程。由于OpenHardwareMonitor初始化需要一定的时间，为了防止启动时程序卡顿，将其放到后台线程中处理\n            InitOpenHardwareLibInThread();\n        }\n    }\n#endif\n\n    //执行测试代码\n#ifdef _DEBUG\n    CTest::Test();\n#endif\n\n    SendSettingsToPlugin();\n\n    CTrafficMonitorDlg dlg;\n    m_pMainWnd = &dlg;\n    INT_PTR nResponse = dlg.DoModal();\n    if (nResponse == IDOK)\n    {\n        // TODO: 在此放置处理何时用\n        //  “确定”来关闭对话框的代码\n    }\n    else if (nResponse == IDCANCEL)\n    {\n        // TODO: 在此放置处理何时用\n        //  “取消”来关闭对话框的代码\n    }\n    else if (nResponse == -1)\n    {\n        TRACE(traceAppMsg, 0, \"警告: 对话框创建失败，应用程序将意外终止。\\n\");\n        TRACE(traceAppMsg, 0, \"警告: 如果您在对话框上使用 MFC 控件，则无法 #define _AFX_NO_MFC_CONTROLS_IN_DIALOGS。\\n\");\n    }\n\n    // 删除上面创建的 shell 管理器。\n    if (pShellManager != NULL)\n    {\n        delete pShellManager;\n    }\n\n#ifndef _AFXDLL\n    ControlBarCleanUp();\n#endif\n\n    // 由于对话框已关闭，所以将返回 FALSE 以便退出应用程序，\n    //  而不是启动应用程序的消息泵。\n    return FALSE;\n}\n\nvoid CTrafficMonitorApp::InitOpenHardwareLibInThread()\n{\n#ifndef WITHOUT_TEMPERATURE\n    AfxBeginThread(InitOpenHardwareMonitorLibThreadFunc, NULL);\n#endif\n}\n\n\nvoid CTrafficMonitorApp::UpdateOpenHardwareMonitorEnableState()\n{\n#ifndef WITHOUT_TEMPERATURE\n    if (m_pMonitor != nullptr)\n    {\n        CSingleLock sync(&theApp.m_minitor_lib_critical, TRUE);\n        m_pMonitor->SetCpuEnable(m_general_data.IsHardwareEnable(HI_CPU));\n        m_pMonitor->SetGpuEnable(m_general_data.IsHardwareEnable(HI_GPU));\n        m_pMonitor->SetHddEnable(m_general_data.IsHardwareEnable(HI_HDD));\n        m_pMonitor->SetMainboardEnable(m_general_data.IsHardwareEnable(HI_MBD));\n    }\n#endif\n}\n\n//void CTrafficMonitorApp::UpdateTaskbarWndMenu()\n//{\n//    //获取“显示设置”子菜单\n//    CMenu* pMenu = m_taskbar_menu.GetSubMenu(0)->GetSubMenu(5);\n//    ASSERT(pMenu != nullptr);\n//    if (pMenu != nullptr)\n//    {\n//        //将ID_SHOW_MEMORY_USAGE后面的所有菜单项删除\n//        if (pMenu->GetMenuItemCount() > 4)\n//        {\n//            int start_pos = CCommon::GetMenuItemPosition(pMenu, ID_SHOW_MEMORY_USAGE) + 1;\n//            while (pMenu->GetMenuItemCount() > start_pos)\n//            {\n//                pMenu->DeleteMenu(start_pos, MF_BYPOSITION);\n//            }\n//        }\n//\n//        //添加温度相关菜单项\n//#ifndef WITHOUT_TEMPERATURE\n//        if (m_general_data.IsHardwareEnable(HI_GPU))\n//            pMenu->AppendMenu(MF_STRING | MF_ENABLED, ID_SHOW_GPU, CCommon::LoadText(IDS_GPU_USAGE));\n//        if (m_general_data.IsHardwareEnable(HI_CPU))\n//            pMenu->AppendMenu(MF_STRING | MF_ENABLED, ID_SHOW_CPU_TEMPERATURE, CCommon::LoadText(IDS_CPU_TEMPERATURE));\n//        if (m_general_data.IsHardwareEnable(HI_GPU))\n//            pMenu->AppendMenu(MF_STRING | MF_ENABLED, ID_SHOW_GPU_TEMPERATURE, CCommon::LoadText(IDS_GPU_TEMPERATURE));\n//        if (m_general_data.IsHardwareEnable(HI_HDD))\n//            pMenu->AppendMenu(MF_STRING | MF_ENABLED, ID_SHOW_HDD_TEMPERATURE, CCommon::LoadText(IDS_HDD_TEMPERATURE));\n//        if (m_general_data.IsHardwareEnable(HI_MBD))\n//            pMenu->AppendMenu(MF_STRING | MF_ENABLED, ID_SHOW_MAIN_BOARD_TEMPERATURE, CCommon::LoadText(IDS_MAINBOARD_TEMPERATURE));\n//        if (m_general_data.IsHardwareEnable(HI_HDD))\n//            pMenu->AppendMenu(MF_STRING | MF_ENABLED, ID_SHOW_HDD, CCommon::LoadText(IDS_HDD_USAGE));\n//#endif\n//\n//        pMenu->AppendMenu(MF_STRING | MF_ENABLED, ID_SHOW_TOTAL_SPEED, CCommon::LoadText(IDS_TOTAL_NET_SPEED));\n//\n//        //添加插件菜单项\n//        if (!m_plugins.GetPluginItems().empty())\n//            pMenu->AppendMenu(MF_SEPARATOR);\n//        int plugin_index = 0;\n//        for (const auto& plugin_item : m_plugins.GetPluginItems())\n//        {\n//            pMenu->AppendMenu(MF_STRING | MF_ENABLED, ID_SHOW_PLUGIN_ITEM_START + plugin_index, plugin_item->GetItemName());\n//            plugin_index++;\n//        }\n//    }\n//}\n\nbool CTrafficMonitorApp::IsForceShowNotifyIcon()\n{\n    return ((!m_cfg_data.m_show_task_bar_wnd /*|| m_win_version.IsWindows11OrLater()*/)\n        && (m_cfg_data.m_hide_main_window || m_main_wnd_data.m_mouse_penetrate));    //如果没有显示任务栏窗口，且隐藏了主窗口或设置了鼠标穿透，则禁用“显示通知区图标”菜单项\n}\n\nstd::wstring CTrafficMonitorApp::GetPlauginTooltipInfo() const\n{\n    std::wstring tip_info;\n    for (const auto& item : m_plugins.GetPlugins())\n    {\n        if (item.plugin != nullptr && item.plugin->GetAPIVersion() >= 2)\n        {\n            std::wstring plugin_tooltip = item.plugin->GetTooltipInfo();\n            if (!plugin_tooltip.empty())\n            {\n                tip_info += L\"\\r\\n\";\n                tip_info += plugin_tooltip.c_str();\n            }\n        }\n    }\n    return tip_info;\n}\n\nbool CTrafficMonitorApp::IsTaksbarItemDisplayed(CommonDisplayItem item) const\n{\n    if (item.is_plugin)\n    {\n        if (item.plugin_item != nullptr)\n            return m_taskbar_data.plugin_display_item.Contains(item.plugin_item->GetItemId());\n    }\n    else\n    {\n        return m_taskbar_data.m_tbar_display_item & item.item_type;\n    }\n    return false;\n}\n\nvoid CTrafficMonitorApp::SendSettingsToPlugin()\n{\n    for (const auto& plugin_info : m_plugins.GetPlugins())\n    {\n        if (plugin_info.plugin != nullptr && plugin_info.plugin->GetAPIVersion() >= 2)\n        {\n            plugin_info.plugin->OnExtenedInfo(ITMPlugin::EI_NAIN_WND_NET_SPEED_SHORT_MODE, std::to_wstring(m_main_wnd_data.speed_short_mode).c_str());\n            plugin_info.plugin->OnExtenedInfo(ITMPlugin::EI_MAIN_WND_SPERATE_WITH_SPACE, std::to_wstring(m_main_wnd_data.separate_value_unit_with_space).c_str());\n            plugin_info.plugin->OnExtenedInfo(ITMPlugin::EI_MAIN_WND_UNIT_BYTE, std::to_wstring(m_main_wnd_data.unit_byte).c_str());\n            plugin_info.plugin->OnExtenedInfo(ITMPlugin::EI_MAIN_WND_UNIT_SELECT, std::to_wstring(static_cast<int>(m_main_wnd_data.speed_unit)).c_str());\n            plugin_info.plugin->OnExtenedInfo(ITMPlugin::EI_MAIN_WND_NOT_SHOW_UNIT, std::to_wstring(m_main_wnd_data.hide_unit).c_str());\n            plugin_info.plugin->OnExtenedInfo(ITMPlugin::EI_MAIN_WND_NOT_SHOW_PERCENT, std::to_wstring(m_main_wnd_data.hide_percent).c_str());\n\n            plugin_info.plugin->OnExtenedInfo(ITMPlugin::EI_TASKBAR_WND_NET_SPEED_SHORT_MODE, std::to_wstring(m_taskbar_data.speed_short_mode).c_str());\n            plugin_info.plugin->OnExtenedInfo(ITMPlugin::EI_TASKBAR_WND_SPERATE_WITH_SPACE, std::to_wstring(m_taskbar_data.separate_value_unit_with_space).c_str());\n            plugin_info.plugin->OnExtenedInfo(ITMPlugin::EI_TASKBAR_WND_VALUE_RIGHT_ALIGN, std::to_wstring(m_taskbar_data.value_right_align).c_str());\n            plugin_info.plugin->OnExtenedInfo(ITMPlugin::EI_TASKBAR_WND_NET_SPEED_WIDTH, std::to_wstring(m_taskbar_data.digits_number).c_str());\n            plugin_info.plugin->OnExtenedInfo(ITMPlugin::EI_TASKBAR_WND_UNIT_BYTE, std::to_wstring(m_taskbar_data.unit_byte).c_str());\n            plugin_info.plugin->OnExtenedInfo(ITMPlugin::EI_TASKBAR_WND_UNIT_SELECT, std::to_wstring(static_cast<int>(m_taskbar_data.speed_unit)).c_str());\n            plugin_info.plugin->OnExtenedInfo(ITMPlugin::EI_TASKBAR_WND_NOT_SHOW_UNIT, std::to_wstring(m_taskbar_data.hide_unit).c_str());\n            plugin_info.plugin->OnExtenedInfo(ITMPlugin::EI_TASKBAR_WND_NOT_SHOW_PERCENT, std::to_wstring(m_taskbar_data.hide_percent).c_str());\n        }\n    }\n}\n\nvoid CTrafficMonitorApp::OnHelp()\n{\n    // TODO: 在此添加命令处理程序代码\n    ShellExecute(NULL, _T(\"open\"), _T(\"https://github.com/zhongyang219/TrafficMonitor/wiki\"), NULL, NULL, SW_SHOW);\n}\n\n\nvoid CTrafficMonitorApp::OnFrequentyAskedQuestions()\n{\n    // TODO: 在此添加命令处理程序代码\n    CString url_domain;\n    if (static_cast<CUpdateHelper::UpdateSource>(m_general_data.update_source) == CUpdateHelper::UpdateSource::GiteeSource)\n        url_domain = _T(\"gitee.com\");\n    else\n        url_domain = _T(\"github.com\");\n    CString language_code{ CCommon::LoadText(IDS_LANGUAGE_CODE) };\n    CString file_name;\n    if (language_code == _T(\"2\"))\n        file_name = _T(\"Help.md\");\n    else\n        file_name = _T(\"Help_en-us.md\");\n    CString url;\n    url.Format(_T(\"https://%s/zhongyang219/TrafficMonitor/blob/master/%s\"), url_domain.GetString(), file_name.GetString());\n    ShellExecute(NULL, _T(\"open\"), url, NULL, NULL, SW_SHOW);\n}\n\n\nvoid CTrafficMonitorApp::OnUpdateLog()\n{\n    // TODO: 在此添加命令处理程序代码\n    CString url_domain;\n    if (static_cast<CUpdateHelper::UpdateSource>(m_general_data.update_source) == CUpdateHelper::UpdateSource::GiteeSource)\n        url_domain = _T(\"gitee.com\");\n    else\n        url_domain = _T(\"github.com\");\n    CString language_code{ CCommon::LoadText(IDS_LANGUAGE_CODE) };\n    CString file_name;\n    if (language_code == _T(\"2\"))\n        file_name = _T(\"update_log.md\");\n    else if (language_code == _T(\"3\"))\n        file_name = _T(\"update_log_zh-tw.md\");\n    else\n        file_name = _T(\"update_log_en-us.md\");\n    CString url;\n    url.Format(_T(\"https://%s/zhongyang219/TrafficMonitor/blob/master/UpdateLog/%s\"), url_domain.GetString(), file_name.GetString());\n    ShellExecute(NULL, _T(\"open\"), url, NULL, NULL, SW_SHOW);\n}\n"
  },
  {
    "path": "TrafficMonitor/TrafficMonitor.h",
    "content": "﻿\n// TrafficMonitor.h : PROJECT_NAME 应用程序的主头文件\n//\n\n#pragma once\n\n#ifndef __AFXWIN_H__\n#error \"在包含此文件之前包含“stdafx.h”以生成 PCH 文件\"\n#endif\n\n#include \"resource.h\"       // 主符号\n#include \"Common.h\"\n#include \"IniHelper.h\"\n#include \"WinVersionHelper.h\"\n#include \"SimpleXML.h\"\n#include \"TaskbarDefaultStyle.h\"\n#include <map>\n#include \"OpenHardwareMonitor/OpenHardwareMonitorApi.h\"\n#include \"PluginManager.h\"\n\n// CTrafficMonitorApp:\n// 有关此类的实现，请参阅 TrafficMonitor.cpp\n//\n\n\nclass CTrafficMonitorApp : public CWinApp\n{\npublic:\n    //各种路径\n    static CTrafficMonitorApp* self;\n    wstring m_module_dir;       //程序exe文件的目录\n    wstring m_appdata_dir;\n    wstring m_module_path;      //程序exe文件的路径\n    wstring m_module_path_reg;  //用于作为写入注册表开机自项的exe文件的路径（如果路径中有空格，加上引号）\n    wstring m_config_path;\n    wstring m_history_traffic_path;\n    wstring m_log_path;\n    wstring m_skin_path;\n    wstring m_system_dir;\n    wstring m_config_dir;\n\n    //以下数据定义为App类中的公共成员，以便于在主对话框和任务栏窗口中都能访问\n    unsigned __int64 m_in_speed{};      //下载速度\n    unsigned __int64 m_out_speed{};     //上传速度\n    int m_cpu_usage{ -1 };      //CPU利用率\n    int m_memory_usage{ -1 };   //内存利用率\n    int m_used_memory{};    //可用物理内存（单位为KB）\n    int m_total_memory{};   //物理内存总量（单位为KB）\n    float m_cpu_temperature{ -1 };  //CPU温度\n    float m_cpu_freq{ -1 };  //CPU 频率\n    float m_gpu_temperature{ -1 };  //显卡温度\n    float m_hdd_temperature{ -1 };  //硬盘温度\n    float m_main_board_temperature{ -1 };    //主板温度\n    int m_gpu_usage{ -1 };      //显卡利用率\n    int m_hdd_usage{ -1 };      //硬盘利用率\n\n    unsigned __int64 m_today_up_traffic{};  //今天已使用的上传流量\n    unsigned __int64 m_today_down_traffic{};    //今天已使用的下载流量\n\n    bool m_cannot_save_config_warning{ true };  //指示是否会在无法保存设置时弹出提示框\n    bool m_cannot_save_global_config_warning{ true };   //指示是否会在无法保存设置时弹出提示框\n\n    bool m_module_dir_writable{ true };         //指示程序所在目录是否可写\n\n    //选项设置数据\n    MainWndSettingData m_main_wnd_data;\n    TaskBarSettingData m_taskbar_data;\n    GeneralSettingData m_general_data;\n    //其他设置数据\n    MainConfigData m_cfg_data;\n    int m_notify_interval;      //弹出通知消息的时间间隔\n    bool m_debug_log{};\n    bool m_taksbar_transparent_color_enable{};\n    bool m_last_light_mode{};\n    bool m_show_mouse_panetrate_tip{};  //是否显示开启“鼠标穿透”时的提示消息。\n    bool m_show_dot_net_notinstalled_tip{};\n    bool m_is_windows11_taskbar{ false };  //是否为Windows11的任务栏\n\n    //bool m_is_windows10_fall_creator;\n    CWinVersionHelper m_win_version;        //当前Windows的版本\n\n    HICON m_notify_icons[MAX_NOTIFY_ICON];\n\n    CTaskbarDefaultStyle m_taskbar_default_style;\n\n    CPluginManager m_plugins;\n\n    CMenu m_main_menu;\n    CMenu m_taskbar_menu;\n\n#ifndef WITHOUT_TEMPERATURE\n    //OpenHardwareMonitor 接口的指针\n    std::shared_ptr<OpenHardwareMonitorApi::IOpenHardwareMonitor> m_pMonitor{};\n#endif // !WITHOUT_TEMPERATURE\n\n    CCriticalSection m_minitor_lib_critical;        //用于访问OpenHardwareMonitor进行线程同步的临界区对象\n    //CCriticalSection m_lftable_critical;            //用于访问LfTable2进行线程同步的临界区对象\n\npublic:\n    CTrafficMonitorApp();\n\n    void LoadConfig();\n    void SaveConfig();\n    void LoadPluginDisabledSettings();\n\n    void LoadGlobalConfig();\n    void SaveGlobalConfig();\n\n    int DPI(int pixel);\n    void DPI(CRect& rect);\n    void DPIFromWindow(CWnd* pWnd);\n    int GetDpi() const { return m_dpi; }\n    void SetDPI(int dpi) { m_dpi = dpi; }\n\n    void CheckUpdate(bool message);     //检查更新，如果message为true，则在检查时弹出提示信息\n    void CheckUpdateInThread(bool message); //在后台线程中检查更新\n    //启动时检查更新线程函数\n    static UINT CheckUpdateThreadFunc(LPVOID lpParam);\n    static UINT InitOpenHardwareMonitorLibThreadFunc(LPVOID lpParam);\n\n    bool SetAutoRun(bool auto_run);\n    bool GetAutoRun(wstring* auto_run_path);        //判断是否开机自动进行，如果是，将开机自动运行的路径写入auto_run_path\n\n    bool SetAutoRunByRegistry(bool auto_run);       //通过注册表实现开机自启动\n    bool SetAutoRunByTaskScheduler(bool auto_run);  //通过任务计划实现开机自启动\n\n    //获取系统信息文本\n    CString GetSystemInfoString();\n\n    void InitMenuResourse();\n\n    //获取一个图标资源，如果资源还未加载，会自动加载。\n    //由于本函数中使用了CTrafficMonitorApp::DPI函数，因此本函数必须确保在CTrafficMonitorApp::DPIFromWindow之后调用\n    HICON GetMenuIcon(UINT id);\n\n    void AutoSelectNotifyIcon();\n\n    bool IsCheckingForUpdate() const { return m_checking_update; }      //是否正在检查更新\n\n    void InitOpenHardwareLibInThread();     //开启一个后台线程初始化OpenHardwareMonitor\n    void UpdateOpenHardwareMonitorEnableState();    //更新硬件监控的启用/禁用状态\n\n    //void UpdateTaskbarWndMenu();      //更新任务栏窗口右键菜单\n    bool IsForceShowNotifyIcon();       //是否需要强制显示通知区图标\n\n    std::wstring GetPlauginTooltipInfo() const;\n    bool IsTaksbarItemDisplayed(CommonDisplayItem item) const;\n\n    void SendSettingsToPlugin();    //向所有插件发送当前的选项设置\n\nprivate:\n    //int m_no_multistart_warning_time{};       //用于设置在开机后多长时间内不弹出“已经有一个程序正在运行”的警告提示\n    bool m_no_multistart_warning{};         //如果为false，则永远都不会弹出“已经有一个程序正在运行”的警告提示\n    bool m_exit_when_start_by_restart_manager{ true };      //如果程序被Windows重启管理器重新启动，则退出程序\n    int m_dpi{ 96 };\n\n    bool m_checking_update{ false };        //是否正在检查更新\n\n    std::map<UINT, HICON> m_menu_icons;      //菜单图标资源。key是图标资源的ID，vlaue是图标的句柄\n\n// 重写\npublic:\n    virtual BOOL InitInstance();\n\n    // 实现\n\n    DECLARE_MESSAGE_MAP()\n    afx_msg void OnHelp();\n    afx_msg void OnFrequentyAskedQuestions();\n    afx_msg void OnUpdateLog();\n};\n\nextern CTrafficMonitorApp theApp;\n"
  },
  {
    "path": "TrafficMonitor/TrafficMonitor.vcxproj",
    "content": "﻿<?xml version=\"1.0\" encoding=\"utf-8\"?>\r\n<Project DefaultTargets=\"Build\" ToolsVersion=\"15.0\" xmlns=\"http://schemas.microsoft.com/developer/msbuild/2003\">\r\n  <ItemGroup Label=\"ProjectConfigurations\">\r\n    <ProjectConfiguration Include=\"Debug (lite)|Win32\">\r\n      <Configuration>Debug (lite)</Configuration>\r\n      <Platform>Win32</Platform>\r\n    </ProjectConfiguration>\r\n    <ProjectConfiguration Include=\"Debug (lite)|x64\">\r\n      <Configuration>Debug (lite)</Configuration>\r\n      <Platform>x64</Platform>\r\n    </ProjectConfiguration>\r\n    <ProjectConfiguration Include=\"Debug|Win32\">\r\n      <Configuration>Debug</Configuration>\r\n      <Platform>Win32</Platform>\r\n    </ProjectConfiguration>\r\n    <ProjectConfiguration Include=\"Release (lite)|Win32\">\r\n      <Configuration>Release (lite)</Configuration>\r\n      <Platform>Win32</Platform>\r\n    </ProjectConfiguration>\r\n    <ProjectConfiguration Include=\"Release (lite)|x64\">\r\n      <Configuration>Release (lite)</Configuration>\r\n      <Platform>x64</Platform>\r\n    </ProjectConfiguration>\r\n    <ProjectConfiguration Include=\"Release|Win32\">\r\n      <Configuration>Release</Configuration>\r\n      <Platform>Win32</Platform>\r\n    </ProjectConfiguration>\r\n    <ProjectConfiguration Include=\"Debug|x64\">\r\n      <Configuration>Debug</Configuration>\r\n      <Platform>x64</Platform>\r\n    </ProjectConfiguration>\r\n    <ProjectConfiguration Include=\"Release|x64\">\r\n      <Configuration>Release</Configuration>\r\n      <Platform>x64</Platform>\r\n    </ProjectConfiguration>\r\n  </ItemGroup>\r\n  <PropertyGroup Label=\"Globals\">\r\n    <ProjectGuid>{09483BED-B1E9-4827-8120-A18302C84AA8}</ProjectGuid>\r\n    <RootNamespace>TrafficMonitor</RootNamespace>\r\n    <WindowsTargetPlatformVersion>10.0</WindowsTargetPlatformVersion>\r\n    <Keyword>MFCProj</Keyword>\r\n  </PropertyGroup>\r\n  <Import Project=\"$(VCTargetsPath)\\Microsoft.Cpp.Default.props\" />\r\n  <PropertyGroup Condition=\"'$(Configuration)|$(Platform)'=='Debug|Win32'\" Label=\"Configuration\">\r\n    <ConfigurationType>Application</ConfigurationType>\r\n    <UseDebugLibraries>true</UseDebugLibraries>\r\n    <PlatformToolset>v142</PlatformToolset>\r\n    <CharacterSet>Unicode</CharacterSet>\r\n    <UseOfMfc>Dynamic</UseOfMfc>\r\n  </PropertyGroup>\r\n  <PropertyGroup Condition=\"'$(Configuration)|$(Platform)'=='Debug (lite)|Win32'\" Label=\"Configuration\">\r\n    <ConfigurationType>Application</ConfigurationType>\r\n    <UseDebugLibraries>true</UseDebugLibraries>\r\n    <PlatformToolset>v142</PlatformToolset>\r\n    <CharacterSet>Unicode</CharacterSet>\r\n    <UseOfMfc>Dynamic</UseOfMfc>\r\n  </PropertyGroup>\r\n  <PropertyGroup Condition=\"'$(Configuration)|$(Platform)'=='Release|Win32'\" Label=\"Configuration\">\r\n    <ConfigurationType>Application</ConfigurationType>\r\n    <UseDebugLibraries>false</UseDebugLibraries>\r\n    <PlatformToolset>v142</PlatformToolset>\r\n    <WholeProgramOptimization>true</WholeProgramOptimization>\r\n    <CharacterSet>Unicode</CharacterSet>\r\n    <UseOfMfc>Dynamic</UseOfMfc>\r\n  </PropertyGroup>\r\n  <PropertyGroup Condition=\"'$(Configuration)|$(Platform)'=='Release (lite)|Win32'\" Label=\"Configuration\">\r\n    <ConfigurationType>Application</ConfigurationType>\r\n    <UseDebugLibraries>false</UseDebugLibraries>\r\n    <PlatformToolset>v142</PlatformToolset>\r\n    <WholeProgramOptimization>true</WholeProgramOptimization>\r\n    <CharacterSet>Unicode</CharacterSet>\r\n    <UseOfMfc>Dynamic</UseOfMfc>\r\n  </PropertyGroup>\r\n  <PropertyGroup Condition=\"'$(Configuration)|$(Platform)'=='Debug|x64'\" Label=\"Configuration\">\r\n    <ConfigurationType>Application</ConfigurationType>\r\n    <UseDebugLibraries>true</UseDebugLibraries>\r\n    <PlatformToolset>v142</PlatformToolset>\r\n    <CharacterSet>Unicode</CharacterSet>\r\n    <UseOfMfc>Dynamic</UseOfMfc>\r\n  </PropertyGroup>\r\n  <PropertyGroup Condition=\"'$(Configuration)|$(Platform)'=='Debug (lite)|x64'\" Label=\"Configuration\">\r\n    <ConfigurationType>Application</ConfigurationType>\r\n    <UseDebugLibraries>true</UseDebugLibraries>\r\n    <PlatformToolset>v142</PlatformToolset>\r\n    <CharacterSet>Unicode</CharacterSet>\r\n    <UseOfMfc>Dynamic</UseOfMfc>\r\n  </PropertyGroup>\r\n  <PropertyGroup Condition=\"'$(Configuration)|$(Platform)'=='Release|x64'\" Label=\"Configuration\">\r\n    <ConfigurationType>Application</ConfigurationType>\r\n    <UseDebugLibraries>false</UseDebugLibraries>\r\n    <PlatformToolset>v142</PlatformToolset>\r\n    <WholeProgramOptimization>true</WholeProgramOptimization>\r\n    <CharacterSet>Unicode</CharacterSet>\r\n    <UseOfMfc>Dynamic</UseOfMfc>\r\n  </PropertyGroup>\r\n  <PropertyGroup Condition=\"'$(Configuration)|$(Platform)'=='Release (lite)|x64'\" Label=\"Configuration\">\r\n    <ConfigurationType>Application</ConfigurationType>\r\n    <UseDebugLibraries>false</UseDebugLibraries>\r\n    <PlatformToolset>v142</PlatformToolset>\r\n    <WholeProgramOptimization>true</WholeProgramOptimization>\r\n    <CharacterSet>Unicode</CharacterSet>\r\n    <UseOfMfc>Dynamic</UseOfMfc>\r\n  </PropertyGroup>\r\n  <Import Project=\"$(VCTargetsPath)\\Microsoft.Cpp.props\" />\r\n  <ImportGroup Label=\"ExtensionSettings\">\r\n  </ImportGroup>\r\n  <ImportGroup Label=\"Shared\">\r\n  </ImportGroup>\r\n  <ImportGroup Label=\"PropertySheets\" Condition=\"'$(Configuration)|$(Platform)'=='Debug|Win32'\">\r\n    <Import Project=\"$(UserRootDir)\\Microsoft.Cpp.$(Platform).user.props\" Condition=\"exists('$(UserRootDir)\\Microsoft.Cpp.$(Platform).user.props')\" Label=\"LocalAppDataPlatform\" />\r\n  </ImportGroup>\r\n  <ImportGroup Condition=\"'$(Configuration)|$(Platform)'=='Debug (lite)|Win32'\" Label=\"PropertySheets\">\r\n    <Import Project=\"$(UserRootDir)\\Microsoft.Cpp.$(Platform).user.props\" Condition=\"exists('$(UserRootDir)\\Microsoft.Cpp.$(Platform).user.props')\" Label=\"LocalAppDataPlatform\" />\r\n  </ImportGroup>\r\n  <ImportGroup Label=\"PropertySheets\" Condition=\"'$(Configuration)|$(Platform)'=='Release|Win32'\">\r\n    <Import Project=\"$(UserRootDir)\\Microsoft.Cpp.$(Platform).user.props\" Condition=\"exists('$(UserRootDir)\\Microsoft.Cpp.$(Platform).user.props')\" Label=\"LocalAppDataPlatform\" />\r\n  </ImportGroup>\r\n  <ImportGroup Condition=\"'$(Configuration)|$(Platform)'=='Release (lite)|Win32'\" Label=\"PropertySheets\">\r\n    <Import Project=\"$(UserRootDir)\\Microsoft.Cpp.$(Platform).user.props\" Condition=\"exists('$(UserRootDir)\\Microsoft.Cpp.$(Platform).user.props')\" Label=\"LocalAppDataPlatform\" />\r\n  </ImportGroup>\r\n  <ImportGroup Label=\"PropertySheets\" Condition=\"'$(Configuration)|$(Platform)'=='Debug|x64'\">\r\n    <Import Project=\"$(UserRootDir)\\Microsoft.Cpp.$(Platform).user.props\" Condition=\"exists('$(UserRootDir)\\Microsoft.Cpp.$(Platform).user.props')\" Label=\"LocalAppDataPlatform\" />\r\n  </ImportGroup>\r\n  <ImportGroup Condition=\"'$(Configuration)|$(Platform)'=='Debug (lite)|x64'\" Label=\"PropertySheets\">\r\n    <Import Project=\"$(UserRootDir)\\Microsoft.Cpp.$(Platform).user.props\" Condition=\"exists('$(UserRootDir)\\Microsoft.Cpp.$(Platform).user.props')\" Label=\"LocalAppDataPlatform\" />\r\n  </ImportGroup>\r\n  <ImportGroup Label=\"PropertySheets\" Condition=\"'$(Configuration)|$(Platform)'=='Release|x64'\">\r\n    <Import Project=\"$(UserRootDir)\\Microsoft.Cpp.$(Platform).user.props\" Condition=\"exists('$(UserRootDir)\\Microsoft.Cpp.$(Platform).user.props')\" Label=\"LocalAppDataPlatform\" />\r\n  </ImportGroup>\r\n  <ImportGroup Condition=\"'$(Configuration)|$(Platform)'=='Release (lite)|x64'\" Label=\"PropertySheets\">\r\n    <Import Project=\"$(UserRootDir)\\Microsoft.Cpp.$(Platform).user.props\" Condition=\"exists('$(UserRootDir)\\Microsoft.Cpp.$(Platform).user.props')\" Label=\"LocalAppDataPlatform\" />\r\n  </ImportGroup>\r\n  <PropertyGroup Label=\"UserMacros\" />\r\n  <PropertyGroup Condition=\"'$(Configuration)|$(Platform)'=='Debug|Win32'\">\r\n    <LinkIncremental>true</LinkIncremental>\r\n    <ExcludePath />\r\n    <OutDir>$(SolutionDir)Bin\\$(Configuration)\\</OutDir>\r\n    <IncludePath>$(ProjectDir)..\\include;$(IncludePath)</IncludePath>\r\n    <LibraryPath>$(SolutionDir)lib\\;$(OutDir);$(LibraryPath)</LibraryPath>\r\n  </PropertyGroup>\r\n  <PropertyGroup Condition=\"'$(Configuration)|$(Platform)'=='Debug (lite)|Win32'\">\r\n    <LinkIncremental>true</LinkIncremental>\r\n    <ExcludePath />\r\n    <OutDir>$(SolutionDir)Bin\\$(Configuration)\\</OutDir>\r\n    <IncludePath>$(ProjectDir)..\\include;$(IncludePath)</IncludePath>\r\n    <LibraryPath>$(SolutionDir)lib\\;$(OutDir);$(LibraryPath)</LibraryPath>\r\n  </PropertyGroup>\r\n  <PropertyGroup Condition=\"'$(Configuration)|$(Platform)'=='Debug|x64'\">\r\n    <LinkIncremental>true</LinkIncremental>\r\n    <ExcludePath />\r\n    <OutDir>$(SolutionDir)Bin\\$(Platform)\\$(Configuration)\\</OutDir>\r\n    <IncludePath>$(ProjectDir)..\\include;$(IncludePath)</IncludePath>\r\n    <LibraryPath>$(SolutionDir)lib\\x64;$(OutDir);$(LibraryPath)</LibraryPath>\r\n  </PropertyGroup>\r\n  <PropertyGroup Condition=\"'$(Configuration)|$(Platform)'=='Debug (lite)|x64'\">\r\n    <LinkIncremental>true</LinkIncremental>\r\n    <ExcludePath />\r\n    <OutDir>$(SolutionDir)Bin\\$(Platform)\\$(Configuration)\\</OutDir>\r\n    <IncludePath>$(ProjectDir)..\\include;$(IncludePath)</IncludePath>\r\n    <LibraryPath>$(SolutionDir)lib\\x64;$(OutDir);$(LibraryPath)</LibraryPath>\r\n  </PropertyGroup>\r\n  <PropertyGroup Condition=\"'$(Configuration)|$(Platform)'=='Release|Win32'\">\r\n    <LinkIncremental>false</LinkIncremental>\r\n    <ExcludePath />\r\n    <OutDir>$(SolutionDir)Bin\\$(Configuration)\\</OutDir>\r\n    <IncludePath>$(ProjectDir)..\\include;$(IncludePath)</IncludePath>\r\n    <LibraryPath>$(SolutionDir)lib\\;$(OutDir);$(LibraryPath)</LibraryPath>\r\n  </PropertyGroup>\r\n  <PropertyGroup Condition=\"'$(Configuration)|$(Platform)'=='Release (lite)|Win32'\">\r\n    <LinkIncremental>false</LinkIncremental>\r\n    <ExcludePath />\r\n    <OutDir>$(SolutionDir)Bin\\$(Configuration)\\</OutDir>\r\n    <IncludePath>$(ProjectDir)..\\include;$(IncludePath)</IncludePath>\r\n    <LibraryPath>$(SolutionDir)lib\\;$(OutDir);$(LibraryPath)</LibraryPath>\r\n  </PropertyGroup>\r\n  <PropertyGroup Condition=\"'$(Configuration)|$(Platform)'=='Release|x64'\">\r\n    <LinkIncremental>false</LinkIncremental>\r\n    <ExcludePath />\r\n    <OutDir>$(SolutionDir)Bin\\$(Platform)\\$(Configuration)\\</OutDir>\r\n    <IncludePath>$(ProjectDir)..\\include;$(IncludePath)</IncludePath>\r\n    <LibraryPath>$(SolutionDir)lib\\x64;$(OutDir);$(LibraryPath)</LibraryPath>\r\n  </PropertyGroup>\r\n  <PropertyGroup Condition=\"'$(Configuration)|$(Platform)'=='Release (lite)|x64'\">\r\n    <LinkIncremental>false</LinkIncremental>\r\n    <ExcludePath />\r\n    <OutDir>$(SolutionDir)Bin\\$(Platform)\\$(Configuration)\\</OutDir>\r\n    <IncludePath>$(ProjectDir)..\\include;$(IncludePath)</IncludePath>\r\n    <LibraryPath>$(SolutionDir)lib\\x64;$(OutDir);$(LibraryPath)</LibraryPath>\r\n  </PropertyGroup>\r\n  <ItemDefinitionGroup Condition=\"'$(Configuration)|$(Platform)'=='Debug|Win32'\">\r\n    <ClCompile>\r\n      <PrecompiledHeader>Use</PrecompiledHeader>\r\n      <WarningLevel>Level3</WarningLevel>\r\n      <Optimization>Disabled</Optimization>\r\n      <PreprocessorDefinitions>WIN32;_WINDOWS;_DEBUG;%(PreprocessorDefinitions)</PreprocessorDefinitions>\r\n      <SDLCheck>true</SDLCheck>\r\n      <TreatWarningAsError>false</TreatWarningAsError>\r\n    </ClCompile>\r\n    <Link>\r\n      <SubSystem>Windows</SubSystem>\r\n      <AdditionalDependencies>OpenHardwareMonitorApi.lib;%(AdditionalDependencies)</AdditionalDependencies>\r\n      <UACExecutionLevel>RequireAdministrator</UACExecutionLevel>\r\n    </Link>\r\n    <Midl>\r\n      <MkTypLibCompatible>false</MkTypLibCompatible>\r\n      <ValidateAllParameters>true</ValidateAllParameters>\r\n      <PreprocessorDefinitions>_DEBUG;%(PreprocessorDefinitions)</PreprocessorDefinitions>\r\n    </Midl>\r\n    <ResourceCompile>\r\n      <Culture>0x0804</Culture>\r\n      <PreprocessorDefinitions>_DEBUG;%(PreprocessorDefinitions)</PreprocessorDefinitions>\r\n      <AdditionalIncludeDirectories>$(IntDir);%(AdditionalIncludeDirectories)</AdditionalIncludeDirectories>\r\n    </ResourceCompile>\r\n    <PreBuildEvent>\r\n      <Command>print_compile_time.bat</Command>\r\n    </PreBuildEvent>\r\n  </ItemDefinitionGroup>\r\n  <ItemDefinitionGroup Condition=\"'$(Configuration)|$(Platform)'=='Debug (lite)|Win32'\">\r\n    <ClCompile>\r\n      <PrecompiledHeader>Use</PrecompiledHeader>\r\n      <WarningLevel>Level3</WarningLevel>\r\n      <Optimization>Disabled</Optimization>\r\n      <PreprocessorDefinitions>WIN32;_WINDOWS;_DEBUG;%(PreprocessorDefinitions);WITHOUT_TEMPERATURE</PreprocessorDefinitions>\r\n      <SDLCheck>true</SDLCheck>\r\n      <TreatWarningAsError>false</TreatWarningAsError>\r\n    </ClCompile>\r\n    <Link>\r\n      <SubSystem>Windows</SubSystem>\r\n      <AdditionalDependencies>%(AdditionalDependencies)</AdditionalDependencies>\r\n    </Link>\r\n    <Midl>\r\n      <MkTypLibCompatible>false</MkTypLibCompatible>\r\n      <ValidateAllParameters>true</ValidateAllParameters>\r\n      <PreprocessorDefinitions>_DEBUG;%(PreprocessorDefinitions)</PreprocessorDefinitions>\r\n    </Midl>\r\n    <ResourceCompile>\r\n      <Culture>0x0804</Culture>\r\n      <PreprocessorDefinitions>_DEBUG;%(PreprocessorDefinitions)</PreprocessorDefinitions>\r\n      <AdditionalIncludeDirectories>$(IntDir);%(AdditionalIncludeDirectories)</AdditionalIncludeDirectories>\r\n    </ResourceCompile>\r\n    <PreBuildEvent>\r\n      <Command>print_compile_time.bat</Command>\r\n    </PreBuildEvent>\r\n  </ItemDefinitionGroup>\r\n  <ItemDefinitionGroup Condition=\"'$(Configuration)|$(Platform)'=='Debug|x64'\">\r\n    <ClCompile>\r\n      <PrecompiledHeader>Use</PrecompiledHeader>\r\n      <WarningLevel>Level3</WarningLevel>\r\n      <Optimization>Disabled</Optimization>\r\n      <PreprocessorDefinitions>_WINDOWS;_DEBUG;%(PreprocessorDefinitions)</PreprocessorDefinitions>\r\n      <SDLCheck>true</SDLCheck>\r\n    </ClCompile>\r\n    <Link>\r\n      <SubSystem>Windows</SubSystem>\r\n      <AdditionalDependencies>OpenHardwareMonitorApi.lib;%(AdditionalDependencies)</AdditionalDependencies>\r\n      <UACExecutionLevel>RequireAdministrator</UACExecutionLevel>\r\n    </Link>\r\n    <Midl>\r\n      <MkTypLibCompatible>false</MkTypLibCompatible>\r\n      <ValidateAllParameters>true</ValidateAllParameters>\r\n      <PreprocessorDefinitions>_DEBUG;%(PreprocessorDefinitions)</PreprocessorDefinitions>\r\n    </Midl>\r\n    <ResourceCompile>\r\n      <Culture>0x0804</Culture>\r\n      <PreprocessorDefinitions>_DEBUG;%(PreprocessorDefinitions)</PreprocessorDefinitions>\r\n      <AdditionalIncludeDirectories>$(IntDir);%(AdditionalIncludeDirectories)</AdditionalIncludeDirectories>\r\n    </ResourceCompile>\r\n    <PreBuildEvent>\r\n      <Command>print_compile_time.bat</Command>\r\n    </PreBuildEvent>\r\n  </ItemDefinitionGroup>\r\n  <ItemDefinitionGroup Condition=\"'$(Configuration)|$(Platform)'=='Debug (lite)|x64'\">\r\n    <ClCompile>\r\n      <PrecompiledHeader>Use</PrecompiledHeader>\r\n      <WarningLevel>Level3</WarningLevel>\r\n      <Optimization>Disabled</Optimization>\r\n      <PreprocessorDefinitions>_WINDOWS;_DEBUG;%(PreprocessorDefinitions);WITHOUT_TEMPERATURE</PreprocessorDefinitions>\r\n      <SDLCheck>true</SDLCheck>\r\n    </ClCompile>\r\n    <Link>\r\n      <SubSystem>Windows</SubSystem>\r\n      <AdditionalDependencies>%(AdditionalDependencies)</AdditionalDependencies>\r\n    </Link>\r\n    <Midl>\r\n      <MkTypLibCompatible>false</MkTypLibCompatible>\r\n      <ValidateAllParameters>true</ValidateAllParameters>\r\n      <PreprocessorDefinitions>_DEBUG;%(PreprocessorDefinitions)</PreprocessorDefinitions>\r\n    </Midl>\r\n    <ResourceCompile>\r\n      <Culture>0x0804</Culture>\r\n      <PreprocessorDefinitions>_DEBUG;%(PreprocessorDefinitions)</PreprocessorDefinitions>\r\n      <AdditionalIncludeDirectories>$(IntDir);%(AdditionalIncludeDirectories)</AdditionalIncludeDirectories>\r\n    </ResourceCompile>\r\n    <PreBuildEvent>\r\n      <Command>print_compile_time.bat</Command>\r\n    </PreBuildEvent>\r\n  </ItemDefinitionGroup>\r\n  <ItemDefinitionGroup Condition=\"'$(Configuration)|$(Platform)'=='Release|Win32'\">\r\n    <ClCompile>\r\n      <WarningLevel>Level3</WarningLevel>\r\n      <PrecompiledHeader>Use</PrecompiledHeader>\r\n      <Optimization>MaxSpeed</Optimization>\r\n      <FunctionLevelLinking>true</FunctionLevelLinking>\r\n      <IntrinsicFunctions>true</IntrinsicFunctions>\r\n      <PreprocessorDefinitions>WIN32;_WINDOWS;NDEBUG;%(PreprocessorDefinitions)</PreprocessorDefinitions>\r\n      <SDLCheck>true</SDLCheck>\r\n      <AdditionalOptions>$(ExternalCompilerOptions) %(AdditionalOptions)</AdditionalOptions>\r\n    </ClCompile>\r\n    <Link>\r\n      <SubSystem>Windows</SubSystem>\r\n      <EnableCOMDATFolding>true</EnableCOMDATFolding>\r\n      <OptimizeReferences>true</OptimizeReferences>\r\n      <AdditionalDependencies>OpenHardwareMonitorApi.lib;%(AdditionalDependencies)</AdditionalDependencies>\r\n      <UACExecutionLevel>RequireAdministrator</UACExecutionLevel>\r\n    </Link>\r\n    <Midl>\r\n      <MkTypLibCompatible>false</MkTypLibCompatible>\r\n      <ValidateAllParameters>true</ValidateAllParameters>\r\n      <PreprocessorDefinitions>NDEBUG;%(PreprocessorDefinitions)</PreprocessorDefinitions>\r\n    </Midl>\r\n    <ResourceCompile>\r\n      <Culture>0x0804</Culture>\r\n      <PreprocessorDefinitions>NDEBUG;%(PreprocessorDefinitions)</PreprocessorDefinitions>\r\n      <AdditionalIncludeDirectories>$(IntDir);%(AdditionalIncludeDirectories)</AdditionalIncludeDirectories>\r\n    </ResourceCompile>\r\n    <PreBuildEvent>\r\n      <Command>print_compile_time.bat</Command>\r\n    </PreBuildEvent>\r\n  </ItemDefinitionGroup>\r\n  <ItemDefinitionGroup Condition=\"'$(Configuration)|$(Platform)'=='Release (lite)|Win32'\">\r\n    <ClCompile>\r\n      <WarningLevel>Level3</WarningLevel>\r\n      <PrecompiledHeader>Use</PrecompiledHeader>\r\n      <Optimization>MaxSpeed</Optimization>\r\n      <FunctionLevelLinking>true</FunctionLevelLinking>\r\n      <IntrinsicFunctions>true</IntrinsicFunctions>\r\n      <PreprocessorDefinitions>WIN32;_WINDOWS;NDEBUG;%(PreprocessorDefinitions);WITHOUT_TEMPERATURE</PreprocessorDefinitions>\r\n      <SDLCheck>true</SDLCheck>\r\n      <AdditionalOptions>$(ExternalCompilerOptions) %(AdditionalOptions)</AdditionalOptions>\r\n    </ClCompile>\r\n    <Link>\r\n      <SubSystem>Windows</SubSystem>\r\n      <EnableCOMDATFolding>true</EnableCOMDATFolding>\r\n      <OptimizeReferences>true</OptimizeReferences>\r\n      <AdditionalDependencies>%(AdditionalDependencies)</AdditionalDependencies>\r\n    </Link>\r\n    <Midl>\r\n      <MkTypLibCompatible>false</MkTypLibCompatible>\r\n      <ValidateAllParameters>true</ValidateAllParameters>\r\n      <PreprocessorDefinitions>NDEBUG;%(PreprocessorDefinitions)</PreprocessorDefinitions>\r\n    </Midl>\r\n    <ResourceCompile>\r\n      <Culture>0x0804</Culture>\r\n      <PreprocessorDefinitions>NDEBUG;%(PreprocessorDefinitions)</PreprocessorDefinitions>\r\n      <AdditionalIncludeDirectories>$(IntDir);%(AdditionalIncludeDirectories)</AdditionalIncludeDirectories>\r\n    </ResourceCompile>\r\n    <PreBuildEvent>\r\n      <Command>print_compile_time.bat</Command>\r\n    </PreBuildEvent>\r\n  </ItemDefinitionGroup>\r\n  <ItemDefinitionGroup Condition=\"'$(Configuration)|$(Platform)'=='Release|x64'\">\r\n    <ClCompile>\r\n      <WarningLevel>Level3</WarningLevel>\r\n      <PrecompiledHeader>Use</PrecompiledHeader>\r\n      <Optimization>MaxSpeed</Optimization>\r\n      <FunctionLevelLinking>true</FunctionLevelLinking>\r\n      <IntrinsicFunctions>true</IntrinsicFunctions>\r\n      <PreprocessorDefinitions>_WINDOWS;NDEBUG;%(PreprocessorDefinitions)</PreprocessorDefinitions>\r\n      <SDLCheck>true</SDLCheck>\r\n    </ClCompile>\r\n    <Link>\r\n      <SubSystem>Windows</SubSystem>\r\n      <EnableCOMDATFolding>true</EnableCOMDATFolding>\r\n      <OptimizeReferences>true</OptimizeReferences>\r\n      <AdditionalDependencies>OpenHardwareMonitorApi.lib;%(AdditionalDependencies)</AdditionalDependencies>\r\n      <UACExecutionLevel>RequireAdministrator</UACExecutionLevel>\r\n    </Link>\r\n    <Midl>\r\n      <MkTypLibCompatible>false</MkTypLibCompatible>\r\n      <ValidateAllParameters>true</ValidateAllParameters>\r\n      <PreprocessorDefinitions>NDEBUG;%(PreprocessorDefinitions)</PreprocessorDefinitions>\r\n    </Midl>\r\n    <ResourceCompile>\r\n      <Culture>0x0804</Culture>\r\n      <PreprocessorDefinitions>NDEBUG;%(PreprocessorDefinitions)</PreprocessorDefinitions>\r\n      <AdditionalIncludeDirectories>$(IntDir);%(AdditionalIncludeDirectories)</AdditionalIncludeDirectories>\r\n    </ResourceCompile>\r\n    <PreBuildEvent>\r\n      <Command>print_compile_time.bat</Command>\r\n    </PreBuildEvent>\r\n  </ItemDefinitionGroup>\r\n  <ItemDefinitionGroup Condition=\"'$(Configuration)|$(Platform)'=='Release (lite)|x64'\">\r\n    <ClCompile>\r\n      <WarningLevel>Level3</WarningLevel>\r\n      <PrecompiledHeader>Use</PrecompiledHeader>\r\n      <Optimization>MaxSpeed</Optimization>\r\n      <FunctionLevelLinking>true</FunctionLevelLinking>\r\n      <IntrinsicFunctions>true</IntrinsicFunctions>\r\n      <PreprocessorDefinitions>_WINDOWS;NDEBUG;%(PreprocessorDefinitions);WITHOUT_TEMPERATURE</PreprocessorDefinitions>\r\n      <SDLCheck>true</SDLCheck>\r\n    </ClCompile>\r\n    <Link>\r\n      <SubSystem>Windows</SubSystem>\r\n      <EnableCOMDATFolding>true</EnableCOMDATFolding>\r\n      <OptimizeReferences>true</OptimizeReferences>\r\n      <AdditionalDependencies>%(AdditionalDependencies)</AdditionalDependencies>\r\n    </Link>\r\n    <Midl>\r\n      <MkTypLibCompatible>false</MkTypLibCompatible>\r\n      <ValidateAllParameters>true</ValidateAllParameters>\r\n      <PreprocessorDefinitions>NDEBUG;%(PreprocessorDefinitions)</PreprocessorDefinitions>\r\n    </Midl>\r\n    <ResourceCompile>\r\n      <Culture>0x0804</Culture>\r\n      <PreprocessorDefinitions>NDEBUG;%(PreprocessorDefinitions)</PreprocessorDefinitions>\r\n      <AdditionalIncludeDirectories>$(IntDir);%(AdditionalIncludeDirectories)</AdditionalIncludeDirectories>\r\n    </ResourceCompile>\r\n    <PreBuildEvent>\r\n      <Command>print_compile_time.bat</Command>\r\n    </PreBuildEvent>\r\n  </ItemDefinitionGroup>\r\n  <ItemGroup>\r\n    <Text Include=\"compile_time.txt\" />\r\n    <Text Include=\"ReadMe.txt\" />\r\n    <Text Include=\"res\\Acknowledgement.txt\" />\r\n    <Text Include=\"res\\Acknowledgement_en.txt\" />\r\n  </ItemGroup>\r\n  <ItemGroup>\r\n    <ClInclude Include=\"..\\include\\PluginInterface.h\" />\r\n    <ClInclude Include=\"AboutDlg.h\" />\r\n    <ClInclude Include=\"AdapterCommon.h\" />\r\n    <ClInclude Include=\"AppAlreadyRuningDlg.h\" />\r\n    <ClInclude Include=\"auto_start_helper.h\" />\r\n    <ClInclude Include=\"BaseDialog.h\" />\r\n    <ClInclude Include=\"CalendarHelper.h\" />\r\n    <ClInclude Include=\"CAutoAdaptSettingsDlg.h\" />\r\n    <ClInclude Include=\"CMFCColorDialogEx.h\" />\r\n    <ClInclude Include=\"ColorSettingListCtrl.h\" />\r\n    <ClInclude Include=\"ComboBox2.h\" />\r\n    <ClInclude Include=\"CPUUsage.h\" />\r\n    <ClInclude Include=\"crashtool.h\" />\r\n    <ClInclude Include=\"CTabCtrlEx.h\" />\r\n    <ClInclude Include=\"CVariant.h\" />\r\n    <ClInclude Include=\"DisplayTextSettingDlg.h\" />\r\n    <ClInclude Include=\"DrawCommonEx.h\" />\r\n    <ClInclude Include=\"FilePathHelper.h\" />\r\n    <ClInclude Include=\"HighResolutionTimer.h\" />\r\n    <ClInclude Include=\"HistoryTrafficCalendarDlg.h\" />\r\n    <ClInclude Include=\"HistoryTrafficFile.h\" />\r\n    <ClInclude Include=\"HistoryTrafficListDlg.h\" />\r\n    <ClInclude Include=\"LinkStatic.h\" />\r\n    <ClInclude Include=\"ColorStatic.h\" />\r\n    <ClInclude Include=\"Common.h\" />\r\n    <ClInclude Include=\"CommonData.h\" />\r\n    <ClInclude Include=\"CSkinPreviewView.h\" />\r\n    <ClInclude Include=\"DonateDlg.h\" />\r\n    <ClInclude Include=\"DrawCommon.h\" />\r\n    <ClInclude Include=\"GeneralSettingsDlg.h\" />\r\n    <ClInclude Include=\"HistoryTrafficDlg.h\" />\r\n    <ClInclude Include=\"IconSelectDlg.h\" />\r\n    <ClInclude Include=\"IniHelper.h\" />\r\n    <ClInclude Include=\"HistoryTrafficListCtrl.h\" />\r\n    <ClInclude Include=\"ListCtrlEx.h\" />\r\n    <ClInclude Include=\"MainWndColorDlg.h\" />\r\n    <ClInclude Include=\"MainWndSettingsDlg.h\" />\r\n    <ClInclude Include=\"MessageDlg.h\" />\r\n    <ClInclude Include=\"NetworkInfoDlg.h\" />\r\n    <ClInclude Include=\"OptionsDlg.h\" />\r\n    <ClInclude Include=\"PictureStatic.h\" />\r\n    <ClInclude Include=\"PluginInfoDlg.h\" />\r\n    <ClInclude Include=\"PluginManager.h\" />\r\n    <ClInclude Include=\"PluginManagerDlg.h\" />\r\n    <ClInclude Include=\"Resource.h\" />\r\n    <ClInclude Include=\"SelectConnectionsDlg.h\" />\r\n    <ClInclude Include=\"SetItemOrderDlg.h\" />\r\n    <ClInclude Include=\"SimpleXML.h\" />\r\n    <ClInclude Include=\"SkinDlg.h\" />\r\n    <ClInclude Include=\"SkinFile.h\" />\r\n    <ClInclude Include=\"SpinEdit.h\" />\r\n    <ClInclude Include=\"StaticEx.h\" />\r\n    <ClInclude Include=\"stdafx.h\" />\r\n    <ClInclude Include=\"TabDlg.h\" />\r\n    <ClInclude Include=\"targetver.h\" />\r\n    <ClInclude Include=\"TaskbarColorDlg.h\" />\r\n    <ClInclude Include=\"TaskbarDefaultStyle.h\" />\r\n    <ClInclude Include=\"TaskBarDlg.h\" />\r\n    <ClInclude Include=\"TaskbarItemOrderHelper.h\" />\r\n    <ClInclude Include=\"TaskBarSettingsDlg.h\" />\r\n    <ClInclude Include=\"Test.h\" />\r\n    <ClInclude Include=\"TinyXml2Helper.h\" />\r\n    <ClInclude Include=\"tinyxml2\\tinyxml2.h\" />\r\n    <ClInclude Include=\"TrafficMonitor.h\" />\r\n    <ClInclude Include=\"TrafficMonitorDlg.h\" />\r\n    <ClInclude Include=\"UpdateHelper.h\" />\r\n    <ClInclude Include=\"WIC.h\" />\r\n    <ClInclude Include=\"WindowsSettingHelper.h\" />\r\n    <ClInclude Include=\"WinVersionHelper.h\" />\r\n  </ItemGroup>\r\n  <ItemGroup>\r\n    <ClCompile Include=\"AboutDlg.cpp\" />\r\n    <ClCompile Include=\"AdapterCommon.cpp\" />\r\n    <ClCompile Include=\"AppAlreadyRuningDlg.cpp\" />\r\n    <ClCompile Include=\"auto_start_helper.cpp\" />\r\n    <ClCompile Include=\"BaseDialog.cpp\" />\r\n    <ClCompile Include=\"CalendarHelper.cpp\" />\r\n    <ClCompile Include=\"CAutoAdaptSettingsDlg.cpp\" />\r\n    <ClCompile Include=\"CMFCColorDialogEx.cpp\" />\r\n    <ClCompile Include=\"ColorSettingListCtrl.cpp\" />\r\n    <ClCompile Include=\"ComboBox2.cpp\" />\r\n    <ClCompile Include=\"CommonData.cpp\" />\r\n    <ClCompile Include=\"CPUUsage.cpp\" />\r\n    <ClCompile Include=\"crashtool.cpp\" />\r\n    <ClCompile Include=\"CTabCtrlEx.cpp\" />\r\n    <ClCompile Include=\"CVariant.cpp\" />\r\n    <ClCompile Include=\"DisplayTextSettingDlg.cpp\" />\r\n    <ClCompile Include=\"DrawCommonEx.cpp\" />\r\n    <ClCompile Include=\"FilePathHelper.cpp\" />\r\n    <ClCompile Include=\"HistoryTrafficCalendarDlg.cpp\" />\r\n    <ClCompile Include=\"HistoryTrafficFile.cpp\" />\r\n    <ClCompile Include=\"HistoryTrafficListDlg.cpp\" />\r\n    <ClCompile Include=\"LinkStatic.cpp\" />\r\n    <ClCompile Include=\"ColorStatic.cpp\" />\r\n    <ClCompile Include=\"Common.cpp\" />\r\n    <ClCompile Include=\"CSkinPreviewView.cpp\" />\r\n    <ClCompile Include=\"DonateDlg.cpp\" />\r\n    <ClCompile Include=\"DrawCommon.cpp\" />\r\n    <ClCompile Include=\"GeneralSettingsDlg.cpp\" />\r\n    <ClCompile Include=\"HistoryTrafficDlg.cpp\" />\r\n    <ClCompile Include=\"IconSelectDlg.cpp\" />\r\n    <ClCompile Include=\"IniHelper.cpp\" />\r\n    <ClCompile Include=\"HistoryTrafficListCtrl.cpp\" />\r\n    <ClCompile Include=\"ListCtrlEx.cpp\" />\r\n    <ClCompile Include=\"MainWndColorDlg.cpp\" />\r\n    <ClCompile Include=\"MainWndSettingsDlg.cpp\" />\r\n    <ClCompile Include=\"MessageDlg.cpp\" />\r\n    <ClCompile Include=\"NetworkInfoDlg.cpp\" />\r\n    <ClCompile Include=\"OptionsDlg.cpp\" />\r\n    <ClCompile Include=\"PictureStatic.cpp\" />\r\n    <ClCompile Include=\"PluginInfoDlg.cpp\" />\r\n    <ClCompile Include=\"PluginManager.cpp\" />\r\n    <ClCompile Include=\"PluginManagerDlg.cpp\" />\r\n    <ClCompile Include=\"SelectConnectionsDlg.cpp\" />\r\n    <ClCompile Include=\"SetItemOrderDlg.cpp\" />\r\n    <ClCompile Include=\"SimpleXML.cpp\" />\r\n    <ClCompile Include=\"SkinDlg.cpp\" />\r\n    <ClCompile Include=\"SkinFile.cpp\" />\r\n    <ClCompile Include=\"SpinEdit.cpp\" />\r\n    <ClCompile Include=\"StaticEx.cpp\" />\r\n    <ClCompile Include=\"stdafx.cpp\">\r\n      <PrecompiledHeader Condition=\"'$(Configuration)|$(Platform)'=='Debug|Win32'\">Create</PrecompiledHeader>\r\n      <PrecompiledHeader Condition=\"'$(Configuration)|$(Platform)'=='Debug (lite)|Win32'\">Create</PrecompiledHeader>\r\n      <PrecompiledHeader Condition=\"'$(Configuration)|$(Platform)'=='Debug|x64'\">Create</PrecompiledHeader>\r\n      <PrecompiledHeader Condition=\"'$(Configuration)|$(Platform)'=='Debug (lite)|x64'\">Create</PrecompiledHeader>\r\n      <PrecompiledHeader Condition=\"'$(Configuration)|$(Platform)'=='Release|Win32'\">Create</PrecompiledHeader>\r\n      <PrecompiledHeader Condition=\"'$(Configuration)|$(Platform)'=='Release (lite)|Win32'\">Create</PrecompiledHeader>\r\n      <PrecompiledHeader Condition=\"'$(Configuration)|$(Platform)'=='Release|x64'\">Create</PrecompiledHeader>\r\n      <PrecompiledHeader Condition=\"'$(Configuration)|$(Platform)'=='Release (lite)|x64'\">Create</PrecompiledHeader>\r\n    </ClCompile>\r\n    <ClCompile Include=\"TabDlg.cpp\" />\r\n    <ClCompile Include=\"TaskbarColorDlg.cpp\" />\r\n    <ClCompile Include=\"TaskbarDefaultStyle.cpp\" />\r\n    <ClCompile Include=\"TaskBarDlg.cpp\" />\r\n    <ClCompile Include=\"TaskbarItemOrderHelper.cpp\" />\r\n    <ClCompile Include=\"TaskBarSettingsDlg.cpp\" />\r\n    <ClCompile Include=\"Test.cpp\" />\r\n    <ClCompile Include=\"TinyXml2Helper.cpp\" />\r\n    <ClCompile Include=\"tinyxml2\\tinyxml2.cpp\">\r\n      <PrecompiledHeader Condition=\"'$(Configuration)|$(Platform)'=='Release|Win32'\">NotUsing</PrecompiledHeader>\r\n      <PrecompiledHeader Condition=\"'$(Configuration)|$(Platform)'=='Debug|Win32'\">NotUsing</PrecompiledHeader>\r\n      <PrecompiledHeader Condition=\"'$(Configuration)|$(Platform)'=='Debug (lite)|Win32'\">NotUsing</PrecompiledHeader>\r\n      <PrecompiledHeader Condition=\"'$(Configuration)|$(Platform)'=='Release (lite)|Win32'\">NotUsing</PrecompiledHeader>\r\n      <PrecompiledHeader Condition=\"'$(Configuration)|$(Platform)'=='Debug|x64'\">NotUsing</PrecompiledHeader>\r\n      <PrecompiledHeader Condition=\"'$(Configuration)|$(Platform)'=='Debug (lite)|x64'\">NotUsing</PrecompiledHeader>\r\n      <PrecompiledHeader Condition=\"'$(Configuration)|$(Platform)'=='Release|x64'\">NotUsing</PrecompiledHeader>\r\n      <PrecompiledHeader Condition=\"'$(Configuration)|$(Platform)'=='Release (lite)|x64'\">NotUsing</PrecompiledHeader>\r\n    </ClCompile>\r\n    <ClCompile Include=\"TrafficMonitor.cpp\" />\r\n    <ClCompile Include=\"TrafficMonitorDlg.cpp\" />\r\n    <ClCompile Include=\"UpdateHelper.cpp\" />\r\n    <ClCompile Include=\"WIC.cpp\" />\r\n    <ClCompile Include=\"WindowsSettingHelper.cpp\" />\r\n    <ClCompile Include=\"WinVersionHelper.cpp\" />\r\n  </ItemGroup>\r\n  <ItemGroup>\r\n    <ResourceCompile Include=\"TrafficMonitor.rc\" />\r\n  </ItemGroup>\r\n  <ItemGroup>\r\n    <None Include=\"..\\LICENSE\" />\r\n    <None Include=\"..\\LICENSE_CN\" />\r\n    <None Include=\"res\\license.bin\" />\r\n    <None Include=\"res\\TrafficMonitor.rc2\" />\r\n  </ItemGroup>\r\n  <ItemGroup>\r\n    <Image Include=\"res\\about_background.bmp\" />\r\n    <Image Include=\"res\\about_background_hd.bmp\" />\r\n    <Image Include=\"res\\bitmap3.bmp\" />\r\n    <Image Include=\"res\\donate.bmp\" />\r\n    <Image Include=\"res\\donate_wechart.bmp\" />\r\n    <Image Include=\"res\\menu_icon\\close.ico\" />\r\n    <Image Include=\"res\\menu_icon\\connection.ico\" />\r\n    <Image Include=\"res\\menu_icon\\exit.ico\" />\r\n    <Image Include=\"res\\menu_icon\\function.ico\" />\r\n    <Image Include=\"res\\menu_icon\\help.ico\" />\r\n    <Image Include=\"res\\menu_icon\\info.ico\" />\r\n    <Image Include=\"res\\menu_icon\\item.ico\" />\r\n    <Image Include=\"res\\menu_icon\\lock.ico\" />\r\n    <Image Include=\"res\\menu_icon\\main_window.ico\" />\r\n    <Image Include=\"res\\menu_icon\\more.ico\" />\r\n    <Image Include=\"res\\menu_icon\\mouse.ico\" />\r\n    <Image Include=\"res\\menu_icon\\notify.ico\" />\r\n    <Image Include=\"res\\menu_icon\\pin.ico\" />\r\n    <Image Include=\"res\\menu_icon\\plugins.ico\" />\r\n    <Image Include=\"res\\menu_icon\\setting.ico\" />\r\n    <Image Include=\"res\\menu_icon\\skn.ico\" />\r\n    <Image Include=\"res\\menu_icon\\statistics.ico\" />\r\n    <Image Include=\"res\\menu_icon\\taskbar_window.ico\" />\r\n    <Image Include=\"res\\menu_icon\\task_manager.ico\" />\r\n    <Image Include=\"res\\notifyicon.ico\" />\r\n    <Image Include=\"res\\notifyicon2.ico\" />\r\n    <Image Include=\"res\\notifyicon3.ico\" />\r\n    <Image Include=\"res\\notifyicon4.ico\" />\r\n    <Image Include=\"res\\notifyicon5.ico\" />\r\n    <Image Include=\"res\\notify_preview.bmp\" />\r\n    <Image Include=\"res\\notify_preview_light.bmp\" />\r\n    <Image Include=\"res\\TrafficMonitor.ico\" />\r\n  </ItemGroup>\r\n  <Import Project=\"$(VCTargetsPath)\\Microsoft.Cpp.targets\" />\r\n  <ImportGroup Label=\"ExtensionTargets\">\r\n  </ImportGroup>\r\n  <ProjectExtensions>\r\n    <VisualStudio>\r\n      <UserProperties RESOURCE_FILE=\"TrafficMonitor.rc\" />\r\n    </VisualStudio>\r\n  </ProjectExtensions>\r\n</Project>"
  },
  {
    "path": "TrafficMonitor/TrafficMonitor.vcxproj.filters",
    "content": "﻿<?xml version=\"1.0\" encoding=\"utf-8\"?>\r\n<Project ToolsVersion=\"4.0\" xmlns=\"http://schemas.microsoft.com/developer/msbuild/2003\">\r\n  <ItemGroup>\r\n    <Filter Include=\"资源文件\">\r\n      <UniqueIdentifier>{67DA6AB6-F800-4c08-8B7A-83BB121AAD01}</UniqueIdentifier>\r\n      <Extensions>rc;ico;cur;bmp;dlg;rc2;rct;bin;rgs;gif;jpg;jpeg;jpe;resx;tiff;tif;png;wav;mfcribbon-ms</Extensions>\r\n    </Filter>\r\n    <Filter Include=\"源文件和头文件\">\r\n      <UniqueIdentifier>{4FC737F1-C7A5-4376-A066-2A32D752A2FF}</UniqueIdentifier>\r\n      <Extensions>cpp;c;cc;cxx;def;odl;idl;hpj;bat;asm;asmx</Extensions>\r\n    </Filter>\r\n    <Filter Include=\"源文件和头文件\\对话框类\">\r\n      <UniqueIdentifier>{296f91e9-59c8-4e1c-9281-b3a5aad240d8}</UniqueIdentifier>\r\n    </Filter>\r\n    <Filter Include=\"源文件和头文件\\派生的控件类\">\r\n      <UniqueIdentifier>{ccda107e-9fc9-493f-991e-b5a9f7dfe2f5}</UniqueIdentifier>\r\n    </Filter>\r\n    <Filter Include=\"源文件和头文件\\派生的控件类\\StaticEx\">\r\n      <UniqueIdentifier>{4745c17a-24c6-46a2-a075-b93654aae464}</UniqueIdentifier>\r\n    </Filter>\r\n    <Filter Include=\"源文件和头文件\\TaskBarDlg\">\r\n      <UniqueIdentifier>{19e2ade9-e461-4885-aa2e-7774b60ebc42}</UniqueIdentifier>\r\n    </Filter>\r\n    <Filter Include=\"源文件和头文件\\TrafficMonitor\">\r\n      <UniqueIdentifier>{3a1e6295-c622-4708-b984-c5e7d05db5df}</UniqueIdentifier>\r\n    </Filter>\r\n    <Filter Include=\"源文件和头文件\\TrafficMonitorDlg\">\r\n      <UniqueIdentifier>{22456594-86cd-4c0e-8ab9-e45cc6d78121}</UniqueIdentifier>\r\n    </Filter>\r\n    <Filter Include=\"源文件和头文件\\stdafx\">\r\n      <UniqueIdentifier>{ee6e7f72-a080-4020-92ff-432b8dcdad2f}</UniqueIdentifier>\r\n    </Filter>\r\n    <Filter Include=\"源文件和头文件\\对话框类\\DonateDlg\">\r\n      <UniqueIdentifier>{9085ed32-5818-45d1-836b-bae3c63952d0}</UniqueIdentifier>\r\n    </Filter>\r\n    <Filter Include=\"源文件和头文件\\对话框类\\选项设置对话框\">\r\n      <UniqueIdentifier>{98909116-8f3b-47d0-8e3f-ef90b29a2acc}</UniqueIdentifier>\r\n    </Filter>\r\n    <Filter Include=\"源文件和头文件\\对话框类\\HistoryTrafficDlg\">\r\n      <UniqueIdentifier>{feb21fbc-174b-4c26-aaa7-9f4b5e6ddd99}</UniqueIdentifier>\r\n    </Filter>\r\n    <Filter Include=\"源文件和头文件\\对话框类\\NetworkInfoDlg\">\r\n      <UniqueIdentifier>{cb7cc4d3-21ca-46e1-ab0e-a35ca66a5603}</UniqueIdentifier>\r\n    </Filter>\r\n    <Filter Include=\"源文件和头文件\\对话框类\\SkinDlg\">\r\n      <UniqueIdentifier>{79053472-785c-4667-a4f9-8b06a83ef04f}</UniqueIdentifier>\r\n    </Filter>\r\n    <Filter Include=\"源文件和头文件\\头文件\">\r\n      <UniqueIdentifier>{93995380-89BD-4b04-88EB-625FBE52EBFB}</UniqueIdentifier>\r\n      <Extensions>h;hh;hpp;hxx;hm;inl;inc;xsd</Extensions>\r\n    </Filter>\r\n    <Filter Include=\"源文件和头文件\\公共的类\">\r\n      <UniqueIdentifier>{5db79e4d-ccae-4347-8e35-bea54fc5a0a7}</UniqueIdentifier>\r\n    </Filter>\r\n    <Filter Include=\"源文件和头文件\\对话框类\\IconSelectDlg\">\r\n      <UniqueIdentifier>{eedf8a29-f4cb-4c66-a019-593c9dfd81ae}</UniqueIdentifier>\r\n    </Filter>\r\n    <Filter Include=\"源文件和头文件\\派生的控件类\\PictureStatic\">\r\n      <UniqueIdentifier>{adb67f17-86da-401c-981c-c6f49a1f52ac}</UniqueIdentifier>\r\n    </Filter>\r\n    <Filter Include=\"源文件和头文件\\公共的类\\Common\">\r\n      <UniqueIdentifier>{7f523bbb-ec3f-4626-b1d8-ad45c2467dd0}</UniqueIdentifier>\r\n    </Filter>\r\n    <Filter Include=\"源文件和头文件\\公共的类\\DrawCommon\">\r\n      <UniqueIdentifier>{c4eace17-d9e5-43bf-8e13-30680c7c80be}</UniqueIdentifier>\r\n    </Filter>\r\n    <Filter Include=\"源文件和头文件\\派生的控件类\\SpinEdit\">\r\n      <UniqueIdentifier>{e63367cf-7c0d-4822-ac30-0f484e65f26d}</UniqueIdentifier>\r\n    </Filter>\r\n    <Filter Include=\"源文件和头文件\\派生的控件类\\TabDlg\">\r\n      <UniqueIdentifier>{5d1d612c-7ef4-498c-9d8f-ad11555faa80}</UniqueIdentifier>\r\n    </Filter>\r\n    <Filter Include=\"源文件和头文件\\派生的控件类\\ColorStatic\">\r\n      <UniqueIdentifier>{1c3d2e37-c54d-429a-ac6f-efe82f02902c}</UniqueIdentifier>\r\n    </Filter>\r\n    <Filter Include=\"源文件和头文件\\公共的类\\IniHelper\">\r\n      <UniqueIdentifier>{974cdadc-b70f-420b-b28b-c09a0686077c}</UniqueIdentifier>\r\n    </Filter>\r\n    <Filter Include=\"源文件和头文件\\对话框类\\颜色设置对话框\">\r\n      <UniqueIdentifier>{66fdf1ef-3bc4-445a-b08e-4cd121d3455a}</UniqueIdentifier>\r\n    </Filter>\r\n    <Filter Include=\"源文件和头文件\\派生的控件类\\LinkStatic\">\r\n      <UniqueIdentifier>{6fce899f-3e92-40c2-a69b-ffcfb4a4c19f}</UniqueIdentifier>\r\n    </Filter>\r\n    <Filter Include=\"源文件和头文件\\公共的类\\CalendarHelper\">\r\n      <UniqueIdentifier>{9fa66474-d19b-4357-9813-e511bb20c722}</UniqueIdentifier>\r\n    </Filter>\r\n    <Filter Include=\"源文件和头文件\\公共的类\\AdapterCommon\">\r\n      <UniqueIdentifier>{75cd7f4d-06b2-467e-9960-994560da6e7b}</UniqueIdentifier>\r\n    </Filter>\r\n    <Filter Include=\"源文件和头文件\\公共的类\\WinVersionHelper\">\r\n      <UniqueIdentifier>{dd5aaacf-0f59-47ca-a738-250b15c6bbbb}</UniqueIdentifier>\r\n    </Filter>\r\n    <Filter Include=\"源文件和头文件\\对话框类\\AboutDlg\">\r\n      <UniqueIdentifier>{1f240f1b-347d-4281-a4cf-18bfebfacb6b}</UniqueIdentifier>\r\n    </Filter>\r\n    <Filter Include=\"源文件和头文件\\公共的类\\SimpleXML\">\r\n      <UniqueIdentifier>{8c35a0e3-e526-4f2f-9bc2-ed74dd37b238}</UniqueIdentifier>\r\n    </Filter>\r\n    <Filter Include=\"源文件和头文件\\对话框类\\颜色设置对话框\\MFCColorDialogEx\">\r\n      <UniqueIdentifier>{dd737159-8653-44d0-bf51-f0e3afad4e5c}</UniqueIdentifier>\r\n    </Filter>\r\n    <Filter Include=\"源文件和头文件\\派生的控件类\\CTabCtrlEx\">\r\n      <UniqueIdentifier>{e5835851-f8d8-47dd-9669-f70b6f424386}</UniqueIdentifier>\r\n    </Filter>\r\n    <Filter Include=\"源文件和头文件\\公共的类\\CrashTool\">\r\n      <UniqueIdentifier>{12448ccc-0fa7-4f50-8261-dcb04f068e39}</UniqueIdentifier>\r\n    </Filter>\r\n    <Filter Include=\"源文件和头文件\\对话框类\\MessageDlg\">\r\n      <UniqueIdentifier>{645d28e7-da22-44f8-8e33-894109dde4b0}</UniqueIdentifier>\r\n    </Filter>\r\n    <Filter Include=\"源文件和头文件\\公共的类\\CVariant\">\r\n      <UniqueIdentifier>{1ca17ad5-07ba-4361-bbeb-fdc5047697fc}</UniqueIdentifier>\r\n    </Filter>\r\n    <Filter Include=\"源文件和头文件\\公共的类\\UpdateHelper\">\r\n      <UniqueIdentifier>{2b0e9f5d-dbce-4ba7-a158-7d9b705fcc02}</UniqueIdentifier>\r\n    </Filter>\r\n    <Filter Include=\"源文件和头文件\\公共的类\\CPUUsage\">\r\n      <UniqueIdentifier>{926ebc77-cc7f-431a-896d-3a4019b41991}</UniqueIdentifier>\r\n    </Filter>\r\n    <Filter Include=\"源文件和头文件\\对话框类\\BaseDialog\">\r\n      <UniqueIdentifier>{6f91f987-8834-4e89-b14e-d3ae15040848}</UniqueIdentifier>\r\n    </Filter>\r\n    <Filter Include=\"源文件和头文件\\公共的类\\TaskbarDefaultStyle\">\r\n      <UniqueIdentifier>{d2389f34-94a7-409a-97d0-47890e629c20}</UniqueIdentifier>\r\n    </Filter>\r\n    <Filter Include=\"源文件和头文件\\对话框类\\选项设置对话框\\AutoAdaptSettingsDlg\">\r\n      <UniqueIdentifier>{2393c651-03a5-467c-a927-a3e99fdc48c4}</UniqueIdentifier>\r\n    </Filter>\r\n    <Filter Include=\"源文件和头文件\\公共的类\\HistoryTrafficFile\">\r\n      <UniqueIdentifier>{5bf9c8c3-2115-4e6d-897c-432303615271}</UniqueIdentifier>\r\n    </Filter>\r\n    <Filter Include=\"测试\">\r\n      <UniqueIdentifier>{df6d5fed-e1e8-4538-b028-f046ac67ad3f}</UniqueIdentifier>\r\n    </Filter>\r\n    <Filter Include=\"源文件和头文件\\公共的类\\WIC\">\r\n      <UniqueIdentifier>{2f6c60c6-eed6-4cad-8a1f-8759efb79d1d}</UniqueIdentifier>\r\n    </Filter>\r\n    <Filter Include=\"源文件和头文件\\对话框类\\选项设置对话框\\DisplayTextSettingDlg\">\r\n      <UniqueIdentifier>{b6c785d6-4031-46d1-b224-4dd87a34404e}</UniqueIdentifier>\r\n    </Filter>\r\n    <Filter Include=\"源文件和头文件\\派生的控件类\\HistoryTrafficListCtrl\">\r\n      <UniqueIdentifier>{74012c97-afb7-4e09-8b11-4e2945c37449}</UniqueIdentifier>\r\n    </Filter>\r\n    <Filter Include=\"源文件和头文件\\派生的控件类\\ListCtrlEx\">\r\n      <UniqueIdentifier>{44872bda-27ce-407a-b5a7-fa9524bfb83c}</UniqueIdentifier>\r\n    </Filter>\r\n    <Filter Include=\"源文件和头文件\\公共的类\\SkinFile\">\r\n      <UniqueIdentifier>{f2a40ca2-c391-46a6-8508-b2f8c30c68a2}</UniqueIdentifier>\r\n    </Filter>\r\n    <Filter Include=\"源文件和头文件\\公共的类\\TinyXml2Helper\">\r\n      <UniqueIdentifier>{ece2a796-fdd6-49e1-987e-11968b3f56a6}</UniqueIdentifier>\r\n    </Filter>\r\n    <Filter Include=\"源文件和头文件\\公共的类\\FilePathHelper\">\r\n      <UniqueIdentifier>{2ee3ad9e-4dc0-4ca4-91b1-2c6225bebd63}</UniqueIdentifier>\r\n    </Filter>\r\n    <Filter Include=\"源文件和头文件\\公共的类\\auto_start_helper\">\r\n      <UniqueIdentifier>{20b159a6-36ec-4d11-97c3-86e7c1501088}</UniqueIdentifier>\r\n    </Filter>\r\n    <Filter Include=\"源文件和头文件\\派生的控件类\\ColorSettingListCtrl\">\r\n      <UniqueIdentifier>{68b8c93d-e1b6-4d3d-a8af-5a19d413a96e}</UniqueIdentifier>\r\n    </Filter>\r\n    <Filter Include=\"源文件和头文件\\派生的控件类\\CComboBox2\">\r\n      <UniqueIdentifier>{abe84b35-de18-4853-9545-65354fb3d22f}</UniqueIdentifier>\r\n    </Filter>\r\n    <Filter Include=\"源文件和头文件\\公共的类\\HighResolutionTimer\">\r\n      <UniqueIdentifier>{c164ccd7-eb7c-465c-8828-d7a38eab573c}</UniqueIdentifier>\r\n    </Filter>\r\n    <Filter Include=\"源文件和头文件\\公共的类\\tinyxml2\">\r\n      <UniqueIdentifier>{24b802e1-bc90-4f4c-8862-0ba6310f05f1}</UniqueIdentifier>\r\n    </Filter>\r\n    <Filter Include=\"源文件和头文件\\公共的类\\TaskbarItemOrderHelper\">\r\n      <UniqueIdentifier>{22cfb8c5-2d79-4f91-add4-fb393e20475e}</UniqueIdentifier>\r\n    </Filter>\r\n    <Filter Include=\"源文件和头文件\\对话框类\\选项设置对话框\\SetItemOrderDlg\">\r\n      <UniqueIdentifier>{bc164584-be21-4a91-9af5-772d61ed9529}</UniqueIdentifier>\r\n    </Filter>\r\n    <Filter Include=\"源文件和头文件\\公共的类\\PluginManager\">\r\n      <UniqueIdentifier>{f7dd3288-085a-4de2-9f83-d8ba2f3eff47}</UniqueIdentifier>\r\n    </Filter>\r\n    <Filter Include=\"源文件和头文件\\对话框类\\PluginManagerDlg\">\r\n      <UniqueIdentifier>{dbacfb68-1e03-4bcb-a08b-e87ba558985f}</UniqueIdentifier>\r\n    </Filter>\r\n    <Filter Include=\"源文件和头文件\\公共的类\\CommonData\">\r\n      <UniqueIdentifier>{554c1c22-3662-416f-9a24-ad76bfb5eecd}</UniqueIdentifier>\r\n    </Filter>\r\n    <Filter Include=\"源文件和头文件\\对话框类\\AppAlreadyRuningDlg\">\r\n      <UniqueIdentifier>{cecf73f6-6479-4ce2-aefe-879c9363ab24}</UniqueIdentifier>\r\n    </Filter>\r\n    <Filter Include=\"源文件和头文件\\对话框类\\选项设置对话框\\SelectConnectionsDlg\">\r\n      <UniqueIdentifier>{8a6deeaa-91ab-4c85-b293-ab1844f3ef82}</UniqueIdentifier>\r\n    </Filter>\r\n    <Filter Include=\"源文件和头文件\\公共的类\\WindowsSettingHelper\">\r\n      <UniqueIdentifier>{46dd548a-c66e-4a37-8767-6f9767125287}</UniqueIdentifier>\r\n    </Filter>\r\n  </ItemGroup>\r\n  <ItemGroup>\r\n    <Text Include=\"ReadMe.txt\" />\r\n    <Text Include=\"res\\Acknowledgement.txt\">\r\n      <Filter>资源文件</Filter>\r\n    </Text>\r\n    <Text Include=\"res\\Acknowledgement_en.txt\" />\r\n    <Text Include=\"compile_time.txt\" />\r\n  </ItemGroup>\r\n  <ItemGroup>\r\n    <ClInclude Include=\"targetver.h\">\r\n      <Filter>源文件和头文件\\头文件</Filter>\r\n    </ClInclude>\r\n    <ClInclude Include=\"Resource.h\">\r\n      <Filter>源文件和头文件\\头文件</Filter>\r\n    </ClInclude>\r\n    <ClInclude Include=\"StaticEx.h\">\r\n      <Filter>源文件和头文件\\派生的控件类\\StaticEx</Filter>\r\n    </ClInclude>\r\n    <ClInclude Include=\"TaskBarDlg.h\">\r\n      <Filter>源文件和头文件\\TaskBarDlg</Filter>\r\n    </ClInclude>\r\n    <ClInclude Include=\"TrafficMonitor.h\">\r\n      <Filter>源文件和头文件\\TrafficMonitor</Filter>\r\n    </ClInclude>\r\n    <ClInclude Include=\"TrafficMonitorDlg.h\">\r\n      <Filter>源文件和头文件\\TrafficMonitorDlg</Filter>\r\n    </ClInclude>\r\n    <ClInclude Include=\"stdafx.h\">\r\n      <Filter>源文件和头文件\\stdafx</Filter>\r\n    </ClInclude>\r\n    <ClInclude Include=\"DonateDlg.h\">\r\n      <Filter>源文件和头文件\\对话框类\\DonateDlg</Filter>\r\n    </ClInclude>\r\n    <ClInclude Include=\"MainWndSettingsDlg.h\">\r\n      <Filter>源文件和头文件\\对话框类\\选项设置对话框</Filter>\r\n    </ClInclude>\r\n    <ClInclude Include=\"TaskBarSettingsDlg.h\">\r\n      <Filter>源文件和头文件\\对话框类\\选项设置对话框</Filter>\r\n    </ClInclude>\r\n    <ClInclude Include=\"OptionsDlg.h\">\r\n      <Filter>源文件和头文件\\对话框类\\选项设置对话框</Filter>\r\n    </ClInclude>\r\n    <ClInclude Include=\"GeneralSettingsDlg.h\">\r\n      <Filter>源文件和头文件\\对话框类\\选项设置对话框</Filter>\r\n    </ClInclude>\r\n    <ClInclude Include=\"HistoryTrafficDlg.h\">\r\n      <Filter>源文件和头文件\\对话框类\\HistoryTrafficDlg</Filter>\r\n    </ClInclude>\r\n    <ClInclude Include=\"NetworkInfoDlg.h\">\r\n      <Filter>源文件和头文件\\对话框类\\NetworkInfoDlg</Filter>\r\n    </ClInclude>\r\n    <ClInclude Include=\"SkinDlg.h\">\r\n      <Filter>源文件和头文件\\对话框类\\SkinDlg</Filter>\r\n    </ClInclude>\r\n    <ClInclude Include=\"IconSelectDlg.h\">\r\n      <Filter>源文件和头文件\\对话框类\\IconSelectDlg</Filter>\r\n    </ClInclude>\r\n    <ClInclude Include=\"PictureStatic.h\">\r\n      <Filter>源文件和头文件\\派生的控件类\\PictureStatic</Filter>\r\n    </ClInclude>\r\n    <ClInclude Include=\"CSkinPreviewView.h\">\r\n      <Filter>源文件和头文件\\对话框类\\SkinDlg</Filter>\r\n    </ClInclude>\r\n    <ClInclude Include=\"Common.h\">\r\n      <Filter>源文件和头文件\\公共的类\\Common</Filter>\r\n    </ClInclude>\r\n    <ClInclude Include=\"DrawCommon.h\">\r\n      <Filter>源文件和头文件\\公共的类\\DrawCommon</Filter>\r\n    </ClInclude>\r\n    <ClInclude Include=\"SpinEdit.h\">\r\n      <Filter>源文件和头文件\\派生的控件类\\SpinEdit</Filter>\r\n    </ClInclude>\r\n    <ClInclude Include=\"TabDlg.h\">\r\n      <Filter>源文件和头文件\\派生的控件类\\TabDlg</Filter>\r\n    </ClInclude>\r\n    <ClInclude Include=\"ColorStatic.h\">\r\n      <Filter>源文件和头文件\\派生的控件类\\ColorStatic</Filter>\r\n    </ClInclude>\r\n    <ClInclude Include=\"IniHelper.h\">\r\n      <Filter>源文件和头文件\\公共的类\\IniHelper</Filter>\r\n    </ClInclude>\r\n    <ClInclude Include=\"MainWndColorDlg.h\">\r\n      <Filter>源文件和头文件\\对话框类\\颜色设置对话框</Filter>\r\n    </ClInclude>\r\n    <ClInclude Include=\"LinkStatic.h\">\r\n      <Filter>源文件和头文件\\派生的控件类\\LinkStatic</Filter>\r\n    </ClInclude>\r\n    <ClInclude Include=\"TaskbarColorDlg.h\">\r\n      <Filter>源文件和头文件\\对话框类\\颜色设置对话框</Filter>\r\n    </ClInclude>\r\n    <ClInclude Include=\"HistoryTrafficListDlg.h\">\r\n      <Filter>源文件和头文件\\对话框类\\HistoryTrafficDlg</Filter>\r\n    </ClInclude>\r\n    <ClInclude Include=\"HistoryTrafficCalendarDlg.h\">\r\n      <Filter>源文件和头文件\\对话框类\\HistoryTrafficDlg</Filter>\r\n    </ClInclude>\r\n    <ClInclude Include=\"CalendarHelper.h\">\r\n      <Filter>源文件和头文件\\公共的类\\CalendarHelper</Filter>\r\n    </ClInclude>\r\n    <ClInclude Include=\"AdapterCommon.h\">\r\n      <Filter>源文件和头文件\\公共的类\\AdapterCommon</Filter>\r\n    </ClInclude>\r\n    <ClInclude Include=\"WinVersionHelper.h\">\r\n      <Filter>源文件和头文件\\公共的类\\WinVersionHelper</Filter>\r\n    </ClInclude>\r\n    <ClInclude Include=\"AboutDlg.h\">\r\n      <Filter>源文件和头文件\\对话框类\\AboutDlg</Filter>\r\n    </ClInclude>\r\n    <ClInclude Include=\"SimpleXML.h\">\r\n      <Filter>源文件和头文件\\公共的类\\SimpleXML</Filter>\r\n    </ClInclude>\r\n    <ClInclude Include=\"CMFCColorDialogEx.h\">\r\n      <Filter>源文件和头文件\\对话框类\\颜色设置对话框\\MFCColorDialogEx</Filter>\r\n    </ClInclude>\r\n    <ClInclude Include=\"CTabCtrlEx.h\">\r\n      <Filter>源文件和头文件\\派生的控件类\\CTabCtrlEx</Filter>\r\n    </ClInclude>\r\n    <ClInclude Include=\"crashtool.h\">\r\n      <Filter>源文件和头文件\\公共的类\\CrashTool</Filter>\r\n    </ClInclude>\r\n    <ClInclude Include=\"MessageDlg.h\">\r\n      <Filter>源文件和头文件\\对话框类\\MessageDlg</Filter>\r\n    </ClInclude>\r\n    <ClInclude Include=\"CVariant.h\">\r\n      <Filter>源文件和头文件\\公共的类\\CVariant</Filter>\r\n    </ClInclude>\r\n    <ClInclude Include=\"DrawCommonEx.h\">\r\n      <Filter>源文件和头文件\\公共的类\\DrawCommon</Filter>\r\n    </ClInclude>\r\n    <ClInclude Include=\"UpdateHelper.h\">\r\n      <Filter>源文件和头文件\\公共的类\\UpdateHelper</Filter>\r\n    </ClInclude>\r\n    <ClInclude Include=\"CPUUsage.h\">\r\n      <Filter>源文件和头文件\\公共的类\\CPUUsage</Filter>\r\n    </ClInclude>\r\n    <ClInclude Include=\"BaseDialog.h\">\r\n      <Filter>源文件和头文件\\对话框类\\BaseDialog</Filter>\r\n    </ClInclude>\r\n    <ClInclude Include=\"TaskbarDefaultStyle.h\">\r\n      <Filter>源文件和头文件\\公共的类\\TaskbarDefaultStyle</Filter>\r\n    </ClInclude>\r\n    <ClInclude Include=\"CAutoAdaptSettingsDlg.h\">\r\n      <Filter>源文件和头文件\\对话框类\\选项设置对话框\\AutoAdaptSettingsDlg</Filter>\r\n    </ClInclude>\r\n    <ClInclude Include=\"HistoryTrafficFile.h\">\r\n      <Filter>源文件和头文件\\公共的类\\HistoryTrafficFile</Filter>\r\n    </ClInclude>\r\n    <ClInclude Include=\"Test.h\">\r\n      <Filter>测试</Filter>\r\n    </ClInclude>\r\n    <ClInclude Include=\"WIC.h\">\r\n      <Filter>源文件和头文件\\公共的类\\WIC</Filter>\r\n    </ClInclude>\r\n    <ClInclude Include=\"DisplayTextSettingDlg.h\">\r\n      <Filter>源文件和头文件\\对话框类\\选项设置对话框\\DisplayTextSettingDlg</Filter>\r\n    </ClInclude>\r\n    <ClInclude Include=\"HistoryTrafficListCtrl.h\">\r\n      <Filter>源文件和头文件\\派生的控件类\\HistoryTrafficListCtrl</Filter>\r\n    </ClInclude>\r\n    <ClInclude Include=\"ListCtrlEx.h\">\r\n      <Filter>源文件和头文件\\派生的控件类\\ListCtrlEx</Filter>\r\n    </ClInclude>\r\n    <ClInclude Include=\"SkinFile.h\">\r\n      <Filter>源文件和头文件\\公共的类\\SkinFile</Filter>\r\n    </ClInclude>\r\n    <ClInclude Include=\"TinyXml2Helper.h\">\r\n      <Filter>源文件和头文件\\公共的类\\TinyXml2Helper</Filter>\r\n    </ClInclude>\r\n    <ClInclude Include=\"FilePathHelper.h\">\r\n      <Filter>源文件和头文件\\公共的类\\FilePathHelper</Filter>\r\n    </ClInclude>\r\n    <ClInclude Include=\"auto_start_helper.h\">\r\n      <Filter>源文件和头文件\\公共的类\\auto_start_helper</Filter>\r\n    </ClInclude>\r\n    <ClInclude Include=\"ColorSettingListCtrl.h\">\r\n      <Filter>源文件和头文件\\派生的控件类\\ColorSettingListCtrl</Filter>\r\n    </ClInclude>\r\n    <ClInclude Include=\"ComboBox2.h\">\r\n      <Filter>源文件和头文件\\派生的控件类\\CComboBox2</Filter>\r\n    </ClInclude>\r\n    <ClInclude Include=\"HighResolutionTimer.h\">\r\n      <Filter>源文件和头文件\\公共的类\\HighResolutionTimer</Filter>\r\n    </ClInclude>\r\n    <ClInclude Include=\"tinyxml2\\tinyxml2.h\">\r\n      <Filter>源文件和头文件\\公共的类\\tinyxml2</Filter>\r\n    </ClInclude>\r\n    <ClInclude Include=\"TaskbarItemOrderHelper.h\">\r\n      <Filter>源文件和头文件\\公共的类\\TaskbarItemOrderHelper</Filter>\r\n    </ClInclude>\r\n    <ClInclude Include=\"SetItemOrderDlg.h\">\r\n      <Filter>源文件和头文件\\对话框类\\选项设置对话框\\SetItemOrderDlg</Filter>\r\n    </ClInclude>\r\n    <ClInclude Include=\"..\\include\\PluginInterface.h\">\r\n      <Filter>源文件和头文件\\头文件</Filter>\r\n    </ClInclude>\r\n    <ClInclude Include=\"PluginManager.h\">\r\n      <Filter>源文件和头文件\\公共的类\\PluginManager</Filter>\r\n    </ClInclude>\r\n    <ClInclude Include=\"PluginManagerDlg.h\">\r\n      <Filter>源文件和头文件\\对话框类\\PluginManagerDlg</Filter>\r\n    </ClInclude>\r\n    <ClInclude Include=\"PluginInfoDlg.h\">\r\n      <Filter>源文件和头文件\\对话框类\\PluginManagerDlg</Filter>\r\n    </ClInclude>\r\n    <ClInclude Include=\"CommonData.h\">\r\n      <Filter>源文件和头文件\\公共的类\\CommonData</Filter>\r\n    </ClInclude>\r\n    <ClInclude Include=\"AppAlreadyRuningDlg.h\">\r\n      <Filter>源文件和头文件\\对话框类\\AppAlreadyRuningDlg</Filter>\r\n    </ClInclude>\r\n    <ClInclude Include=\"SelectConnectionsDlg.h\">\r\n      <Filter>源文件和头文件\\对话框类\\选项设置对话框\\SelectConnectionsDlg</Filter>\r\n    </ClInclude>\r\n    <ClInclude Include=\"WindowsSettingHelper.h\">\r\n      <Filter>源文件和头文件\\公共的类\\WindowsSettingHelper</Filter>\r\n    </ClInclude>\r\n  </ItemGroup>\r\n  <ItemGroup>\r\n    <ClCompile Include=\"StaticEx.cpp\">\r\n      <Filter>源文件和头文件\\派生的控件类\\StaticEx</Filter>\r\n    </ClCompile>\r\n    <ClCompile Include=\"TaskBarDlg.cpp\">\r\n      <Filter>源文件和头文件\\TaskBarDlg</Filter>\r\n    </ClCompile>\r\n    <ClCompile Include=\"TrafficMonitor.cpp\">\r\n      <Filter>源文件和头文件\\TrafficMonitor</Filter>\r\n    </ClCompile>\r\n    <ClCompile Include=\"TrafficMonitorDlg.cpp\">\r\n      <Filter>源文件和头文件\\TrafficMonitorDlg</Filter>\r\n    </ClCompile>\r\n    <ClCompile Include=\"stdafx.cpp\">\r\n      <Filter>源文件和头文件\\stdafx</Filter>\r\n    </ClCompile>\r\n    <ClCompile Include=\"DonateDlg.cpp\">\r\n      <Filter>源文件和头文件\\对话框类\\DonateDlg</Filter>\r\n    </ClCompile>\r\n    <ClCompile Include=\"GeneralSettingsDlg.cpp\">\r\n      <Filter>源文件和头文件\\对话框类\\选项设置对话框</Filter>\r\n    </ClCompile>\r\n    <ClCompile Include=\"MainWndSettingsDlg.cpp\">\r\n      <Filter>源文件和头文件\\对话框类\\选项设置对话框</Filter>\r\n    </ClCompile>\r\n    <ClCompile Include=\"TaskBarSettingsDlg.cpp\">\r\n      <Filter>源文件和头文件\\对话框类\\选项设置对话框</Filter>\r\n    </ClCompile>\r\n    <ClCompile Include=\"OptionsDlg.cpp\">\r\n      <Filter>源文件和头文件\\对话框类\\选项设置对话框</Filter>\r\n    </ClCompile>\r\n    <ClCompile Include=\"HistoryTrafficDlg.cpp\">\r\n      <Filter>源文件和头文件\\对话框类\\HistoryTrafficDlg</Filter>\r\n    </ClCompile>\r\n    <ClCompile Include=\"NetworkInfoDlg.cpp\">\r\n      <Filter>源文件和头文件\\对话框类\\NetworkInfoDlg</Filter>\r\n    </ClCompile>\r\n    <ClCompile Include=\"SkinDlg.cpp\">\r\n      <Filter>源文件和头文件\\对话框类\\SkinDlg</Filter>\r\n    </ClCompile>\r\n    <ClCompile Include=\"IconSelectDlg.cpp\">\r\n      <Filter>源文件和头文件\\对话框类\\IconSelectDlg</Filter>\r\n    </ClCompile>\r\n    <ClCompile Include=\"PictureStatic.cpp\">\r\n      <Filter>源文件和头文件\\派生的控件类\\PictureStatic</Filter>\r\n    </ClCompile>\r\n    <ClCompile Include=\"CSkinPreviewView.cpp\">\r\n      <Filter>源文件和头文件\\对话框类\\SkinDlg</Filter>\r\n    </ClCompile>\r\n    <ClCompile Include=\"Common.cpp\">\r\n      <Filter>源文件和头文件\\公共的类\\Common</Filter>\r\n    </ClCompile>\r\n    <ClCompile Include=\"DrawCommon.cpp\">\r\n      <Filter>源文件和头文件\\公共的类\\DrawCommon</Filter>\r\n    </ClCompile>\r\n    <ClCompile Include=\"SpinEdit.cpp\">\r\n      <Filter>源文件和头文件\\派生的控件类\\SpinEdit</Filter>\r\n    </ClCompile>\r\n    <ClCompile Include=\"TabDlg.cpp\">\r\n      <Filter>源文件和头文件\\派生的控件类\\TabDlg</Filter>\r\n    </ClCompile>\r\n    <ClCompile Include=\"ColorStatic.cpp\">\r\n      <Filter>源文件和头文件\\派生的控件类\\ColorStatic</Filter>\r\n    </ClCompile>\r\n    <ClCompile Include=\"IniHelper.cpp\">\r\n      <Filter>源文件和头文件\\公共的类\\IniHelper</Filter>\r\n    </ClCompile>\r\n    <ClCompile Include=\"MainWndColorDlg.cpp\">\r\n      <Filter>源文件和头文件\\对话框类\\颜色设置对话框</Filter>\r\n    </ClCompile>\r\n    <ClCompile Include=\"LinkStatic.cpp\">\r\n      <Filter>源文件和头文件\\派生的控件类\\LinkStatic</Filter>\r\n    </ClCompile>\r\n    <ClCompile Include=\"TaskbarColorDlg.cpp\">\r\n      <Filter>源文件和头文件\\对话框类\\颜色设置对话框</Filter>\r\n    </ClCompile>\r\n    <ClCompile Include=\"HistoryTrafficListDlg.cpp\">\r\n      <Filter>源文件和头文件\\对话框类\\HistoryTrafficDlg</Filter>\r\n    </ClCompile>\r\n    <ClCompile Include=\"HistoryTrafficCalendarDlg.cpp\">\r\n      <Filter>源文件和头文件\\对话框类\\HistoryTrafficDlg</Filter>\r\n    </ClCompile>\r\n    <ClCompile Include=\"CalendarHelper.cpp\">\r\n      <Filter>源文件和头文件\\公共的类\\CalendarHelper</Filter>\r\n    </ClCompile>\r\n    <ClCompile Include=\"AdapterCommon.cpp\">\r\n      <Filter>源文件和头文件\\公共的类\\AdapterCommon</Filter>\r\n    </ClCompile>\r\n    <ClCompile Include=\"WinVersionHelper.cpp\">\r\n      <Filter>源文件和头文件\\公共的类\\WinVersionHelper</Filter>\r\n    </ClCompile>\r\n    <ClCompile Include=\"AboutDlg.cpp\">\r\n      <Filter>源文件和头文件\\对话框类\\AboutDlg</Filter>\r\n    </ClCompile>\r\n    <ClCompile Include=\"SimpleXML.cpp\">\r\n      <Filter>源文件和头文件\\公共的类\\SimpleXML</Filter>\r\n    </ClCompile>\r\n    <ClCompile Include=\"CMFCColorDialogEx.cpp\">\r\n      <Filter>源文件和头文件\\对话框类\\颜色设置对话框\\MFCColorDialogEx</Filter>\r\n    </ClCompile>\r\n    <ClCompile Include=\"CTabCtrlEx.cpp\">\r\n      <Filter>源文件和头文件\\派生的控件类\\CTabCtrlEx</Filter>\r\n    </ClCompile>\r\n    <ClCompile Include=\"crashtool.cpp\">\r\n      <Filter>源文件和头文件\\公共的类\\CrashTool</Filter>\r\n    </ClCompile>\r\n    <ClCompile Include=\"MessageDlg.cpp\">\r\n      <Filter>源文件和头文件\\对话框类\\MessageDlg</Filter>\r\n    </ClCompile>\r\n    <ClCompile Include=\"CVariant.cpp\">\r\n      <Filter>源文件和头文件\\公共的类\\CVariant</Filter>\r\n    </ClCompile>\r\n    <ClCompile Include=\"DrawCommonEx.cpp\">\r\n      <Filter>源文件和头文件\\公共的类\\DrawCommon</Filter>\r\n    </ClCompile>\r\n    <ClCompile Include=\"UpdateHelper.cpp\">\r\n      <Filter>源文件和头文件\\公共的类\\UpdateHelper</Filter>\r\n    </ClCompile>\r\n    <ClCompile Include=\"CPUUsage.cpp\">\r\n      <Filter>源文件和头文件\\公共的类\\CPUUsage</Filter>\r\n    </ClCompile>\r\n    <ClCompile Include=\"BaseDialog.cpp\">\r\n      <Filter>源文件和头文件\\对话框类\\BaseDialog</Filter>\r\n    </ClCompile>\r\n    <ClCompile Include=\"TaskbarDefaultStyle.cpp\">\r\n      <Filter>源文件和头文件\\公共的类\\TaskbarDefaultStyle</Filter>\r\n    </ClCompile>\r\n    <ClCompile Include=\"CAutoAdaptSettingsDlg.cpp\">\r\n      <Filter>源文件和头文件\\对话框类\\选项设置对话框\\AutoAdaptSettingsDlg</Filter>\r\n    </ClCompile>\r\n    <ClCompile Include=\"HistoryTrafficFile.cpp\">\r\n      <Filter>源文件和头文件\\公共的类\\HistoryTrafficFile</Filter>\r\n    </ClCompile>\r\n    <ClCompile Include=\"Test.cpp\">\r\n      <Filter>测试</Filter>\r\n    </ClCompile>\r\n    <ClCompile Include=\"WIC.cpp\">\r\n      <Filter>源文件和头文件\\公共的类\\WIC</Filter>\r\n    </ClCompile>\r\n    <ClCompile Include=\"DisplayTextSettingDlg.cpp\">\r\n      <Filter>源文件和头文件\\对话框类\\选项设置对话框\\DisplayTextSettingDlg</Filter>\r\n    </ClCompile>\r\n    <ClCompile Include=\"HistoryTrafficListCtrl.cpp\">\r\n      <Filter>源文件和头文件\\派生的控件类\\HistoryTrafficListCtrl</Filter>\r\n    </ClCompile>\r\n    <ClCompile Include=\"ListCtrlEx.cpp\">\r\n      <Filter>源文件和头文件\\派生的控件类\\ListCtrlEx</Filter>\r\n    </ClCompile>\r\n    <ClCompile Include=\"SkinFile.cpp\">\r\n      <Filter>源文件和头文件\\公共的类\\SkinFile</Filter>\r\n    </ClCompile>\r\n    <ClCompile Include=\"TinyXml2Helper.cpp\">\r\n      <Filter>源文件和头文件\\公共的类\\TinyXml2Helper</Filter>\r\n    </ClCompile>\r\n    <ClCompile Include=\"FilePathHelper.cpp\">\r\n      <Filter>源文件和头文件\\公共的类\\FilePathHelper</Filter>\r\n    </ClCompile>\r\n    <ClCompile Include=\"auto_start_helper.cpp\">\r\n      <Filter>源文件和头文件\\公共的类\\auto_start_helper</Filter>\r\n    </ClCompile>\r\n    <ClCompile Include=\"ColorSettingListCtrl.cpp\">\r\n      <Filter>源文件和头文件\\派生的控件类\\ColorSettingListCtrl</Filter>\r\n    </ClCompile>\r\n    <ClCompile Include=\"ComboBox2.cpp\">\r\n      <Filter>源文件和头文件\\派生的控件类\\CComboBox2</Filter>\r\n    </ClCompile>\r\n    <ClCompile Include=\"tinyxml2\\tinyxml2.cpp\">\r\n      <Filter>源文件和头文件\\公共的类\\tinyxml2</Filter>\r\n    </ClCompile>\r\n    <ClCompile Include=\"TaskbarItemOrderHelper.cpp\">\r\n      <Filter>源文件和头文件\\公共的类\\TaskbarItemOrderHelper</Filter>\r\n    </ClCompile>\r\n    <ClCompile Include=\"SetItemOrderDlg.cpp\">\r\n      <Filter>源文件和头文件\\对话框类\\选项设置对话框\\SetItemOrderDlg</Filter>\r\n    </ClCompile>\r\n    <ClCompile Include=\"PluginManager.cpp\">\r\n      <Filter>源文件和头文件\\公共的类\\PluginManager</Filter>\r\n    </ClCompile>\r\n    <ClCompile Include=\"PluginManagerDlg.cpp\">\r\n      <Filter>源文件和头文件\\对话框类\\PluginManagerDlg</Filter>\r\n    </ClCompile>\r\n    <ClCompile Include=\"PluginInfoDlg.cpp\">\r\n      <Filter>源文件和头文件\\对话框类\\PluginManagerDlg</Filter>\r\n    </ClCompile>\r\n    <ClCompile Include=\"CommonData.cpp\">\r\n      <Filter>源文件和头文件\\公共的类\\CommonData</Filter>\r\n    </ClCompile>\r\n    <ClCompile Include=\"AppAlreadyRuningDlg.cpp\">\r\n      <Filter>源文件和头文件\\对话框类\\AppAlreadyRuningDlg</Filter>\r\n    </ClCompile>\r\n    <ClCompile Include=\"SelectConnectionsDlg.cpp\">\r\n      <Filter>源文件和头文件\\对话框类\\选项设置对话框\\SelectConnectionsDlg</Filter>\r\n    </ClCompile>\r\n    <ClCompile Include=\"WindowsSettingHelper.cpp\">\r\n      <Filter>源文件和头文件\\公共的类\\WindowsSettingHelper</Filter>\r\n    </ClCompile>\r\n  </ItemGroup>\r\n  <ItemGroup>\r\n    <ResourceCompile Include=\"TrafficMonitor.rc\">\r\n      <Filter>资源文件</Filter>\r\n    </ResourceCompile>\r\n  </ItemGroup>\r\n  <ItemGroup>\r\n    <None Include=\"res\\TrafficMonitor.rc2\">\r\n      <Filter>资源文件</Filter>\r\n    </None>\r\n    <None Include=\"..\\LICENSE\" />\r\n    <None Include=\"res\\license.bin\">\r\n      <Filter>资源文件</Filter>\r\n    </None>\r\n    <None Include=\"..\\LICENSE_CN\" />\r\n  </ItemGroup>\r\n  <ItemGroup>\r\n    <Image Include=\"res\\TrafficMonitor.ico\">\r\n      <Filter>资源文件</Filter>\r\n    </Image>\r\n    <Image Include=\"res\\about_background.bmp\">\r\n      <Filter>资源文件</Filter>\r\n    </Image>\r\n    <Image Include=\"res\\donate.bmp\">\r\n      <Filter>资源文件</Filter>\r\n    </Image>\r\n    <Image Include=\"res\\notifyicon.ico\">\r\n      <Filter>资源文件</Filter>\r\n    </Image>\r\n    <Image Include=\"res\\about_background_hd.bmp\">\r\n      <Filter>资源文件</Filter>\r\n    </Image>\r\n    <Image Include=\"res\\notifyicon2.ico\">\r\n      <Filter>资源文件</Filter>\r\n    </Image>\r\n    <Image Include=\"res\\notifyicon3.ico\">\r\n      <Filter>资源文件</Filter>\r\n    </Image>\r\n    <Image Include=\"res\\notify_preview.bmp\">\r\n      <Filter>资源文件</Filter>\r\n    </Image>\r\n    <Image Include=\"res\\notifyicon4.ico\">\r\n      <Filter>资源文件</Filter>\r\n    </Image>\r\n    <Image Include=\"res\\notify_preview_light.bmp\">\r\n      <Filter>资源文件</Filter>\r\n    </Image>\r\n    <Image Include=\"res\\donate_wechart.bmp\">\r\n      <Filter>资源文件</Filter>\r\n    </Image>\r\n    <Image Include=\"res\\bitmap3.bmp\">\r\n      <Filter>资源文件</Filter>\r\n    </Image>\r\n    <Image Include=\"res\\menu_icon\\exit.ico\">\r\n      <Filter>资源文件</Filter>\r\n    </Image>\r\n    <Image Include=\"res\\menu_icon\\help.ico\">\r\n      <Filter>资源文件</Filter>\r\n    </Image>\r\n    <Image Include=\"res\\menu_icon\\info.ico\">\r\n      <Filter>资源文件</Filter>\r\n    </Image>\r\n    <Image Include=\"res\\menu_icon\\lock.ico\">\r\n      <Filter>资源文件</Filter>\r\n    </Image>\r\n    <Image Include=\"res\\menu_icon\\setting.ico\">\r\n      <Filter>资源文件</Filter>\r\n    </Image>\r\n    <Image Include=\"res\\menu_icon\\statistics.ico\">\r\n      <Filter>资源文件</Filter>\r\n    </Image>\r\n    <Image Include=\"res\\menu_icon\\close.ico\">\r\n      <Filter>资源文件</Filter>\r\n    </Image>\r\n    <Image Include=\"res\\menu_icon\\pin.ico\">\r\n      <Filter>资源文件</Filter>\r\n    </Image>\r\n    <Image Include=\"res\\menu_icon\\skn.ico\">\r\n      <Filter>资源文件</Filter>\r\n    </Image>\r\n    <Image Include=\"res\\menu_icon\\connection.ico\">\r\n      <Filter>资源文件</Filter>\r\n    </Image>\r\n    <Image Include=\"res\\menu_icon\\mouse.ico\">\r\n      <Filter>资源文件</Filter>\r\n    </Image>\r\n    <Image Include=\"res\\menu_icon\\main_window.ico\">\r\n      <Filter>资源文件</Filter>\r\n    </Image>\r\n    <Image Include=\"res\\menu_icon\\taskbar_window.ico\">\r\n      <Filter>资源文件</Filter>\r\n    </Image>\r\n    <Image Include=\"res\\menu_icon\\notify.ico\">\r\n      <Filter>资源文件</Filter>\r\n    </Image>\r\n    <Image Include=\"res\\menu_icon\\more.ico\">\r\n      <Filter>资源文件</Filter>\r\n    </Image>\r\n    <Image Include=\"res\\menu_icon\\item.ico\">\r\n      <Filter>资源文件</Filter>\r\n    </Image>\r\n    <Image Include=\"res\\menu_icon\\function.ico\">\r\n      <Filter>资源文件</Filter>\r\n    </Image>\r\n    <Image Include=\"res\\notifyicon5.ico\">\r\n      <Filter>资源文件</Filter>\r\n    </Image>\r\n    <Image Include=\"res\\menu_icon\\plugins.ico\">\r\n      <Filter>资源文件</Filter>\r\n    </Image>\r\n    <Image Include=\"res\\menu_icon\\task_manager.ico\">\r\n      <Filter>资源文件</Filter>\r\n    </Image>\r\n  </ItemGroup>\r\n</Project>"
  },
  {
    "path": "TrafficMonitor/TrafficMonitorDlg.cpp",
    "content": "﻿\n// TrafficMonitorDlg.cpp : 实现文件\n//\n\n#include \"stdafx.h\"\n#include \"TrafficMonitor.h\"\n#include \"TrafficMonitorDlg.h\"\n#include \"afxdialogex.h\"\n#include \"Test.h\"\n#include \"PluginManagerDlg.h\"\n#include \"SetItemOrderDlg.h\"\n#include \"WindowsSettingHelper.h\"\n\n#ifdef _DEBUG\n#define new DEBUG_NEW\n#endif\n\n\n\n// CTrafficMonitorDlg 对话框\n\n//静态成员初始化\nunsigned int CTrafficMonitorDlg::m_WM_TASKBARCREATED{ ::RegisterWindowMessage(_T(\"TaskbarCreated\")) };  //注册任务栏建立的消息\n\nCTrafficMonitorDlg::CTrafficMonitorDlg(CWnd* pParent /*=NULL*/)\n    : CDialog(IDD_TRAFFICMONITOR_DIALOG, pParent)\n{\n    m_hIcon = AfxGetApp()->LoadIcon(IDR_MAINFRAME);\n    m_desktop_dc = ::GetDC(NULL);\n}\n\nCTrafficMonitorDlg::~CTrafficMonitorDlg()\n{\n    free(m_pIfTable);\n\n    if (m_tBarDlg != nullptr)\n    {\n        delete m_tBarDlg;\n        m_tBarDlg = nullptr;\n    }\n\n    ::ReleaseDC(NULL, m_desktop_dc);\n}\n\nvoid CTrafficMonitorDlg::DoDataExchange(CDataExchange* pDX)\n{\n    CDialog::DoDataExchange(pDX);\n}\n\nBEGIN_MESSAGE_MAP(CTrafficMonitorDlg, CDialog)\n    ON_WM_QUERYDRAGICON()\n    ON_WM_TIMER()\n    ON_WM_RBUTTONUP()\n    ON_WM_LBUTTONDOWN()\n    ON_COMMAND(ID_NETWORK_INFO, &CTrafficMonitorDlg::OnNetworkInfo)\n    ON_COMMAND(ID_ALWAYS_ON_TOP, &CTrafficMonitorDlg::OnAlwaysOnTop)\n    ON_WM_INITMENUPOPUP()\n    ON_COMMAND(ID_TRANSPARENCY_100, &CTrafficMonitorDlg::OnTransparency100)\n    ON_COMMAND(ID_TRANSPARENCY_80, &CTrafficMonitorDlg::OnTransparency80)\n    ON_COMMAND(ID_TRANSPARENCY_60, &CTrafficMonitorDlg::OnTransparency60)\n    ON_COMMAND(ID_TRANSPARENCY_40, &CTrafficMonitorDlg::OnTransparency40)\n    ON_WM_CLOSE()\n    ON_WM_INITMENU()\n    ON_COMMAND(ID_LOCK_WINDOW_POS, &CTrafficMonitorDlg::OnLockWindowPos)\n    ON_WM_MOVE()\n    ON_MESSAGE(MY_WM_NOTIFYICON, &CTrafficMonitorDlg::OnNotifyIcon)\n    ON_COMMAND(ID_SHOW_NOTIFY_ICON, &CTrafficMonitorDlg::OnShowNotifyIcon)\n    ON_WM_DESTROY()\n    ON_COMMAND(ID_SHOW_CPU_MEMORY, &CTrafficMonitorDlg::OnShowCpuMemory)\n    ON_COMMAND(ID_MOUSE_PENETRATE, &CTrafficMonitorDlg::OnMousePenetrate)\n    ON_COMMAND(ID_SHOW_TASK_BAR_WND, &CTrafficMonitorDlg::OnShowTaskBarWnd)\n    ON_COMMAND(ID_APP_ABOUT, &CTrafficMonitorDlg::OnAppAbout)\n    ON_COMMAND(ID_SHOW_CPU_MEMORY2, &CTrafficMonitorDlg::OnShowCpuMemory2)\n    ON_COMMAND(ID_SHOW_MAIN_WND, &CTrafficMonitorDlg::OnShowMainWnd)\n    ON_COMMAND(ID_CHANGE_SKIN, &CTrafficMonitorDlg::OnChangeSkin)\n    ON_REGISTERED_MESSAGE(m_WM_TASKBARCREATED, &CTrafficMonitorDlg::OnTaskBarCreated)\n    ON_COMMAND(ID_TRAFFIC_HISTORY, &CTrafficMonitorDlg::OnTrafficHistory)\n    ON_WM_MOUSEMOVE()\n    ON_WM_LBUTTONDBLCLK()\n    ON_COMMAND(ID_OPTIONS, &CTrafficMonitorDlg::OnOptions)\n    ON_COMMAND(ID_OPTIONS2, &CTrafficMonitorDlg::OnOptions2)\n    ON_MESSAGE(WM_EXITMENULOOP, &CTrafficMonitorDlg::OnExitmenuloop)\n    ON_COMMAND(ID_CHANGE_NOTIFY_ICON, &CTrafficMonitorDlg::OnChangeNotifyIcon)\n    ON_COMMAND(ID_ALOW_OUT_OF_BORDER, &CTrafficMonitorDlg::OnAlowOutOfBorder)\n    ON_COMMAND(ID_CHECK_UPDATE, &CTrafficMonitorDlg::OnCheckUpdate)\n    ON_MESSAGE(WM_TASKBAR_MENU_POPED_UP, &CTrafficMonitorDlg::OnTaskbarMenuPopedUp)\n    ON_COMMAND(ID_SHOW_NET_SPEED, &CTrafficMonitorDlg::OnShowNetSpeed)\n    ON_WM_QUERYENDSESSION()\n    ON_WM_PAINT()\n    ON_MESSAGE(WM_DPICHANGED, &CTrafficMonitorDlg::OnDpichanged)\n    ON_MESSAGE(WM_TASKBAR_WND_CLOSED, &CTrafficMonitorDlg::OnTaskbarWndClosed)\n    ON_MESSAGE(WM_MONITOR_INFO_UPDATED, &CTrafficMonitorDlg::OnMonitorInfoUpdated)\n    ON_MESSAGE(WM_DISPLAYCHANGE, &CTrafficMonitorDlg::OnDisplaychange)\n    ON_WM_EXITSIZEMOVE()\n    ON_COMMAND(ID_PLUGIN_MANAGE, &CTrafficMonitorDlg::OnPluginManage)\n    ON_MESSAGE(WM_REOPEN_TASKBAR_WND, &CTrafficMonitorDlg::OnReopenTaksbarWnd)\n    ON_COMMAND(ID_OPEN_TASK_MANAGER, &CTrafficMonitorDlg::OnOpenTaskManager)\n    ON_MESSAGE(WM_SETTINGS_APPLIED, &CTrafficMonitorDlg::OnSettingsApplied)\n    ON_COMMAND(ID_DISPLAY_SETTINGS, &CTrafficMonitorDlg::OnDisplaySettings)\n    ON_WM_LBUTTONUP()\n    ON_COMMAND(ID_REFRESH_CONNECTION_LIST, &CTrafficMonitorDlg::OnRefreshConnectionList)\nEND_MESSAGE_MAP()\n\n\nCString CTrafficMonitorDlg::GetMouseTipsInfo()\n{\n    CString tip_info;\n    CString temp;\n    temp.Format(_T(\"%s: %s\\r\\n (%s: %s/%s: %s)\"), CCommon::LoadText(IDS_TRAFFIC_USED_TODAY),\n        CCommon::KBytesToString((theApp.m_today_up_traffic + theApp.m_today_down_traffic) / 1024u),\n        CCommon::LoadText(IDS_UPLOAD), CCommon::KBytesToString(theApp.m_today_up_traffic / 1024u),\n        CCommon::LoadText(IDS_DOWNLOAD), CCommon::KBytesToString(theApp.m_today_down_traffic / 1024u)\n    );\n    tip_info += temp;\n    const CSkinFile::Layout& skin_layout{ theApp.m_cfg_data.m_show_more_info ? m_skin.GetLayoutInfo().layout_l : m_skin.GetLayoutInfo().layout_s }; //当前的皮肤布局\n    if (!skin_layout.GetItem(TDI_UP).show)      //如果主窗口中没有显示上传速度，则在提示信息中显示上传速度\n    {\n        temp.Format(_T(\"\\r\\n%s: %s/s\"), CCommon::LoadText(IDS_UPLOAD),\n            CCommon::DataSizeToString(theApp.m_out_speed, theApp.m_main_wnd_data));\n        tip_info += temp;\n    }\n    if (!skin_layout.GetItem(TDI_DOWN).show)\n    {\n        temp.Format(_T(\"\\r\\n%s: %s/s\"), CCommon::LoadText(IDS_DOWNLOAD),\n            CCommon::DataSizeToString(theApp.m_in_speed, theApp.m_main_wnd_data));\n        tip_info += temp;\n    }\n    if (!skin_layout.GetItem(TDI_CPU).show)\n    {\n        temp.Format(_T(\"\\r\\n%s: %d %%\"), CCommon::LoadText(IDS_CPU_USAGE), theApp.m_cpu_usage);\n        tip_info += temp;\n    }\n    if (!skin_layout.GetItem(TDI_MEMORY).show)\n    {\n        temp.Format(_T(\"\\r\\n%s: %s/%s (%d %%)\"), CCommon::LoadText(IDS_MEMORY_USAGE),\n            CCommon::KBytesToString(theApp.m_used_memory),\n            CCommon::KBytesToString(theApp.m_total_memory), theApp.m_memory_usage);\n        tip_info += temp;\n    }\n    else\n    {\n        temp.Format(_T(\"\\r\\n%s: %s/%s\"), CCommon::LoadText(IDS_MEMORY_USAGE),\n            CCommon::KBytesToString(theApp.m_used_memory),\n            CCommon::KBytesToString(theApp.m_total_memory));\n        tip_info += temp;\n    }\n#ifndef WITHOUT_TEMPERATURE\n    if (IsTemperatureNeeded())\n    {\n        if (theApp.m_general_data.IsHardwareEnable(HI_GPU) && !skin_layout.GetItem(TDI_GPU_USAGE).show && theApp.m_gpu_usage >= 0)\n        {\n            temp.Format(_T(\"\\r\\n%s: %d %%\"), CCommon::LoadText(IDS_GPU_USAGE), theApp.m_gpu_usage);\n            tip_info += temp;\n        }\n        if (theApp.m_general_data.IsHardwareEnable(HI_GPU) && !skin_layout.GetItem(TDI_CPU_FREQ).show && theApp.m_cpu_freq >= 0)\n        {\n            temp.Format(_T(\"\\r\\n%s: %d %%\"), CCommon::LoadText(IDS_CPU_FREQ), theApp.m_cpu_freq);\n            tip_info += temp;\n        }\n        if (theApp.m_general_data.IsHardwareEnable(HI_CPU) && !skin_layout.GetItem(TDI_CPU_TEMP).show && theApp.m_cpu_temperature > 0)\n        {\n            temp.Format(_T(\"\\r\\n%s: %s\"), CCommon::LoadText(IDS_CPU_TEMPERATURE), CCommon::TemperatureToString(theApp.m_cpu_temperature, theApp.m_main_wnd_data));\n            tip_info += temp;\n        }\n        if (theApp.m_general_data.IsHardwareEnable(HI_GPU) && !skin_layout.GetItem(TDI_GPU_TEMP).show && theApp.m_gpu_temperature > 0)\n        {\n            temp.Format(_T(\"\\r\\n%s: %s\"), CCommon::LoadText(IDS_GPU_TEMPERATURE), CCommon::TemperatureToString(theApp.m_gpu_temperature, theApp.m_main_wnd_data));\n            tip_info += temp;\n        }\n        if (theApp.m_general_data.IsHardwareEnable(HI_HDD) && !skin_layout.GetItem(TDI_HDD_TEMP).show && theApp.m_hdd_temperature > 0)\n        {\n            temp.Format(_T(\"\\r\\n%s: %s\"), CCommon::LoadText(IDS_HDD_TEMPERATURE), CCommon::TemperatureToString(theApp.m_hdd_temperature, theApp.m_main_wnd_data));\n            tip_info += temp;\n        }\n        if (theApp.m_general_data.IsHardwareEnable(HI_MBD) && !skin_layout.GetItem(TDI_MAIN_BOARD_TEMP).show && theApp.m_main_board_temperature > 0)\n        {\n            temp.Format(_T(\"\\r\\n%s: %s\"), CCommon::LoadText(IDS_MAINBOARD_TEMPERATURE), CCommon::TemperatureToString(theApp.m_main_board_temperature, theApp.m_main_wnd_data));\n            tip_info += temp;\n        }\n        if (theApp.m_general_data.IsHardwareEnable(HI_HDD) && !skin_layout.GetItem(TDI_HDD_USAGE).show && theApp.m_hdd_usage >= 0)\n        {\n            temp.Format(_T(\"\\r\\n%s: %d %%\"), CCommon::LoadText(IDS_HDD_USAGE), theApp.m_hdd_usage);\n            tip_info += temp;\n        }\n    }\n#endif\n    //添加插件项目的鼠标提示\n    tip_info += theApp.GetPlauginTooltipInfo().c_str();\n\n    return tip_info;\n}\n\nvoid CTrafficMonitorDlg::SetTransparency()\n{\n    SetWindowLong(m_hWnd, GWL_EXSTYLE, GetWindowLong(m_hWnd, GWL_EXSTYLE) | WS_EX_LAYERED);\n    SetLayeredWindowAttributes(0, theApp.m_cfg_data.m_transparency * 255 / 100, LWA_ALPHA);  //透明度取值范围为0~255\n}\n\nvoid CTrafficMonitorDlg::SetTransparency(int transparency)\n{\n    SetWindowLong(m_hWnd, GWL_EXSTYLE, GetWindowLong(m_hWnd, GWL_EXSTYLE) | WS_EX_LAYERED);\n    SetLayeredWindowAttributes(0, transparency * 255 / 100, LWA_ALPHA);  //透明度取值范围为0~255\n}\n\nvoid CTrafficMonitorDlg::SetAlwaysOnTop()\n{\n    //if (!m_is_foreground_fullscreen || (m_is_foreground_fullscreen && !theApp.m_main_wnd_data.hide_main_wnd_when_fullscreen))\n    //{\n    //  if (m_always_on_top)\n    //      SetWindowPos(&wndTopMost, 0, 0, 0, 0, SWP_NOSIZE | SWP_NOMOVE);         //设置置顶\n    //  else\n    //      SetWindowPos(&wndNoTopMost, 0, 0, 0, 0, SWP_NOSIZE | SWP_NOMOVE);       //取消置顶\n    //}\n    if (theApp.m_cfg_data.m_hide_main_window)\n        return;\n    else if (theApp.m_main_wnd_data.hide_main_wnd_when_fullscreen && m_is_foreground_fullscreen)        //当设置有程序全屏时隐藏悬浮窗且有程序在全屏运行时，不执行置顶操作\n        return;\n\n    if (theApp.m_main_wnd_data.m_always_on_top)\n        SetWindowPos(&wndTopMost, 0, 0, 0, 0, SWP_NOSIZE | SWP_NOMOVE);         //设置置顶\n    else\n        SetWindowPos(&wndNoTopMost, 0, 0, 0, 0, SWP_NOSIZE | SWP_NOMOVE);       //取消置顶\n\n}\n\nvoid CTrafficMonitorDlg::SetMousePenetrate()\n{\n    if (theApp.m_main_wnd_data.m_mouse_penetrate)\n    {\n        SetWindowLong(m_hWnd, GWL_EXSTYLE, GetWindowLong(m_hWnd, GWL_EXSTYLE) | WS_EX_TRANSPARENT);     //设置鼠标穿透\n    }\n    else\n    {\n        SetWindowLong(m_hWnd, GWL_EXSTYLE, GetWindowLong(m_hWnd, GWL_EXSTYLE) & (~WS_EX_TRANSPARENT));      //取消鼠标穿透\n    }\n}\n\nPOINT CTrafficMonitorDlg::CalculateWindowMoveOffset(CRect rect, bool screen_changed)\n{\n    POINT mov{};    // 所需偏移量\n    if (m_screen_rects.size() != 0)\n    {\n        // 确保窗口完整在一个监视器内并且可见，判断移动距离并向所需移动距离较小的方向移动\n        LONG mov_xy = 0;          // 记录移动距离\n        int i = 0;\n        for (auto& a : m_screen_rects)\n        {\n            LONG x = 0, y = 0;\n            if (rect.left < a.left)                 // 需要向右移动\n                x = a.left - rect.left;\n            if (rect.top < a.top)                   // 需要向下移动\n                y = a.top - rect.top;\n\n            //防止连接新屏幕时 m_last_screen_rects 的数量小于 m_screen_rects 的数量导致访问到过大的索引\n            if (i >= m_last_screen_rects.size())\n            {\n                break;\n            }\n            CRect last_screen_rect = m_last_screen_rects[i];\n            if (screen_changed && a != last_screen_rect)\n            {\n                float proportion_width = (float)rect.left / (last_screen_rect.right - rect.Width());\n                float proportion_height = (float)rect.top / (last_screen_rect.bottom - rect.Height());\n                x = (a.right - rect.Width()) * proportion_width - rect.left;\n                y = (a.bottom - rect.Height()) * proportion_height - rect.top;\n            }\n            else\n            {\n                if (rect.right > a.right)          // 需要向左移动\n                    x = a.right - rect.right;\n                if (rect.bottom > a.bottom)        // 需要向上移动\n                    y = a.bottom - rect.bottom;\n            }\n            if (x == 0 && y == 0)           // 窗口已在一个监视器内\n            {\n                mov.x = 0;\n                mov.y = 0;\n                break;\n            }\n            else if (abs(x) + abs(y) < mov_xy || mov_xy == 0)\n            {\n                mov.x = x;\n                mov.y = y;\n                mov_xy = abs(x) + abs(y);\n            }\n            i++;\n        }\n    }\n    return mov;\n}\n\nvoid CTrafficMonitorDlg::CheckWindowPos(bool screen_changed)\n{\n    if (!theApp.m_main_wnd_data.m_alow_out_of_border)\n    {\n        CRect rect;\n        GetWindowRect(rect);\n        MoveWindow(rect + CalculateWindowMoveOffset(rect, screen_changed));\n    }\n}\n\nvoid CTrafficMonitorDlg::GetScreenSize()\n{\n    m_screen_size.cx = GetSystemMetrics(SM_CXSCREEN);\n    m_screen_size.cy = GetSystemMetrics(SM_CYSCREEN);\n\n    //::SystemParametersInfo(SPI_GETWORKAREA, 0, &m_screen_rect, 0);   // 获得工作区大小\n\n    //获取所有屏幕工作区的大小\n    m_last_screen_rects = m_screen_rects;\n    m_screen_rects.clear();\n    Monitors monitors;\n    for (auto& a : monitors.monitorinfos)\n    {\n        m_screen_rects.push_back(a.rcWork);\n    }\n}\n\n\nvoid CTrafficMonitorDlg::AutoSelect()\n{\n    unsigned __int64 max_in_out_bytes{};\n    unsigned __int64 in_out_bytes;\n    //m_connection_selected = m_connections[0].index;\n    m_connection_selected = 0;\n    //自动选择连接时，查找已发送和已接收字节数之和最多的那个连接，并将其设置为当前查看的连接\n    for (size_t i{}; i < m_connections.size(); i++)\n    {\n        auto table = GetConnectIfTable(i);\n        if (table.dwOperStatus == IF_OPER_STATUS_OPERATIONAL)     //只选择网络状态为正常的连接\n        {\n            in_out_bytes = table.dwInOctets + table.dwOutOctets;\n            if (in_out_bytes > max_in_out_bytes)\n            {\n                max_in_out_bytes = in_out_bytes;\n                m_connection_selected = i;\n            }\n        }\n    }\n    theApp.m_cfg_data.m_connection_name = GetConnection(m_connection_selected).description_2;\n    m_connection_change_flag = true;\n}\n\nvoid CTrafficMonitorDlg::IniConnection()\n{\n    //为m_pIfTable开辟所需大小的内存\n    free(m_pIfTable);\n    m_dwSize = sizeof(MIB_IFTABLE);\n    m_pIfTable = (MIB_IFTABLE*)malloc(m_dwSize);\n    int rtn;\n    rtn = GetIfTable(m_pIfTable, &m_dwSize, FALSE);\n    if (rtn == ERROR_INSUFFICIENT_BUFFER)\t//如果函数返回值为ERROR_INSUFFICIENT_BUFFER，说明m_pIfTable的大小不够\n    {\n        free(m_pIfTable);\n        m_pIfTable = (MIB_IFTABLE*)malloc(m_dwSize);\t//用新的大小重新开辟一块内存\n    }\n    GetIfTable(m_pIfTable, &m_dwSize, FALSE);\n\n    //获取当前所有的连接，并保存到m_connections容器中\n    if (!theApp.m_general_data.show_all_interface)\n    {\n        m_connections.clear();\n        vector<NetWorkConection> connections;\n        CAdapterCommon::GetAdapterInfo(connections);\n        for (const auto& item : connections)\n        {\n            if (!theApp.m_general_data.connections_hide.Contains(CCommon::StrToUnicode(item.description.c_str())))\n                m_connections.push_back(item);\n        }\n        CAdapterCommon::GetIfTableInfo(m_connections, m_pIfTable);\n    }\n    else\n    {\n        CAdapterCommon::GetAllIfTableInfo(m_connections, m_pIfTable);\n    }\n\n    //如果在设置了“显示所有网络连接”时设置了“选择全部”，则改为“自动选择”\n    if (theApp.m_general_data.show_all_interface && theApp.m_cfg_data.m_select_all)\n    {\n        theApp.m_cfg_data.m_select_all = false;\n        theApp.m_cfg_data.m_auto_select = true;\n    }\n\n    //写入调试日志\n    if (theApp.m_debug_log)\n    {\n        CString log_str;\n        log_str += _T(\"正在初始化网络连接...\\n\");\n        log_str += _T(\"连接列表：\\n\");\n        for (size_t i{}; i < m_connections.size(); i++)\n        {\n            log_str += m_connections[i].description.c_str();\n            log_str += _T(\", \");\n            log_str += CCommon::IntToString(m_connections[i].index);\n            log_str += _T(\"\\n\");\n        }\n        log_str += _T(\"IfTable:\\n\");\n        for (size_t i{}; i < m_pIfTable->dwNumEntries; i++)\n        {\n            log_str += CCommon::IntToString(i);\n            log_str += _T(\" \");\n            log_str += (const char*)m_pIfTable->table[i].bDescr;\n            log_str += _T(\"\\n\");\n        }\n        CCommon::WriteLog(log_str, (theApp.m_config_dir + L\".\\\\connections.log\").c_str());\n    }\n\n    //if (m_connection_selected < 0 || m_connection_selected >= m_connections.size() || m_auto_select)\n    //  AutoSelect();\n    //选择网络连接\n    if (theApp.m_cfg_data.m_auto_select)    //自动选择\n    {\n        if (m_restart_cnt != -1)    //当m_restart_cnt不等于-1时，即不是第一次初始化时，需要延时5秒再重新初始化连接\n        {\n            KillTimer(DELAY_TIMER);\n            SetTimer(DELAY_TIMER, 5000, NULL);\n        }\n        else\n        {\n            AutoSelect();\n        }\n    }\n    else        //查找网络名为上次选择的连接\n    {\n        m_connection_selected = 0;\n        for (size_t i{}; i < m_connections.size(); i++)\n        {\n            if (m_connections[i].description_2 == m_connection_name_preferd)\n                m_connection_selected = i;\n        }\n    }\n    if (m_connection_selected < 0 || m_connection_selected >= m_connections.size())\n        m_connection_selected = 0;\n    theApp.m_cfg_data.m_connection_name = GetConnection(m_connection_selected).description_2;\n\n    //根据已获取到的连接在菜单中添加相应项目\n    CMenu* select_connection_menu = theApp.m_main_menu.GetSubMenu(0)->GetSubMenu(0);        //设置“选择网络连接”子菜单项\n    IniConnectionMenu(select_connection_menu);      //向“选择网卡”子菜单项添加项目\n\n    IniTaskBarConnectionMenu();     //初始化任务栏窗口中的“选择网络连接”子菜单项\n\n    m_restart_cnt++;    //记录初始化次数\n    m_connection_change_flag = true;\n}\n\nMIB_IFROW CTrafficMonitorDlg::GetConnectIfTable(int connection_index)\n{\n    if (connection_index >= 0 && connection_index < static_cast<int>(m_connections.size()))\n    {\n        int index = m_connections[connection_index].index;\n        if (m_pIfTable != nullptr && index >= 0 && index < m_pIfTable->dwNumEntries)\n            return m_pIfTable->table[index];\n    }\n    return MIB_IFROW();\n}\n\nNetWorkConection CTrafficMonitorDlg::GetConnection(int connection_index)\n{\n    if (connection_index >= 0 && connection_index < static_cast<int>(m_connections.size()))\n        return m_connections[connection_index];\n    else\n        return NetWorkConection();\n}\n\nvoid CTrafficMonitorDlg::IniConnectionMenu(CMenu* pMenu)\n{\n    ASSERT(pMenu != nullptr);\n    if (pMenu != nullptr)\n    {\n        //先将ID_SELECT_ALL_CONNECTION后面的所有菜单项删除\n        int start_pos = CCommon::GetMenuItemPosition(pMenu, ID_SELECT_ALL_CONNECTION) + 1;\n        while (pMenu->GetMenuItemCount() > start_pos)\n        {\n            pMenu->DeleteMenu(start_pos, MF_BYPOSITION);\n        }\n\n        CString connection_descr;\n        for (size_t i{}; i < m_connections.size(); i++)\n        {\n            connection_descr = CCommon::StrToUnicode(m_connections[i].description.c_str()).c_str();\n            pMenu->AppendMenu(MF_STRING | MF_ENABLED, ID_SELECT_ALL_CONNECTION + i + 1, connection_descr);\n        }\n\n        //添加“刷新网络列表”命令\n        pMenu->AppendMenu(MF_SEPARATOR);\n        pMenu->AppendMenu(MF_STRING | MF_ENABLED, ID_REFRESH_CONNECTION_LIST, CCommon::LoadText(IDS_REFRESH_CONNECTION_LIST));\n    }\n}\n\nvoid CTrafficMonitorDlg::IniTaskBarConnectionMenu()\n{\n    CMenu* select_connection_menu = theApp.m_taskbar_menu.GetSubMenu(0)->GetSubMenu(0);     //设置“选择网络连接”子菜单项\n    IniConnectionMenu(select_connection_menu);      //向“选择网卡”子菜单项添加项目\n}\n\nvoid CTrafficMonitorDlg::SetConnectionMenuState(CMenu* pMenu)\n{\n    if (theApp.m_cfg_data.m_select_all)\n        pMenu->CheckMenuRadioItem(0, m_connections.size() + 1, 1, MF_BYPOSITION | MF_CHECKED);\n    else if (theApp.m_cfg_data.m_auto_select)       //m_auto_select为true时为自动选择，选中菜单的第1项\n        pMenu->CheckMenuRadioItem(0, m_connections.size() + 1, 0, MF_BYPOSITION | MF_CHECKED);\n    else        //m_auto_select为false时非自动选择，根据m_connection_selected的值选择对应的项\n        pMenu->CheckMenuRadioItem(0, m_connections.size() + 1, m_connection_selected + 2, MF_BYPOSITION | MF_CHECKED);\n}\n\nvoid CTrafficMonitorDlg::CloseTaskBarWnd()\n{\n    if (m_tBarDlg != nullptr)\n    {\n        if (IsTaskbarWndValid())\n            m_tBarDlg->OnCancel();\n        delete m_tBarDlg;\n        m_tBarDlg = nullptr;\n    }\n}\n\nvoid CTrafficMonitorDlg::OpenTaskBarWnd()\n{\n    m_tBarDlg = new CTaskBarDlg;\n    m_tBarDlg->Create(IDD_TASK_BAR_DIALOG, this);\n    m_tBarDlg->ShowWindow(SW_SHOW);\n    //m_tBarDlg->ShowInfo();\n    //IniTaskBarConnectionMenu();\n}\n\nvoid CTrafficMonitorDlg::AddNotifyIcon()\n{\n    if (theApp.m_cfg_data.m_show_task_bar_wnd)\n        CloseTaskBarWnd();\n    //添加通知栏图标\n    ::Shell_NotifyIcon(NIM_ADD, &m_ntIcon);\n    if (theApp.m_cfg_data.m_show_task_bar_wnd)\n        OpenTaskBarWnd();\n}\n\nvoid CTrafficMonitorDlg::DeleteNotifyIcon()\n{\n    if (theApp.m_cfg_data.m_show_task_bar_wnd)\n        CloseTaskBarWnd();\n    //删除通知栏图标\n    ::Shell_NotifyIcon(NIM_DELETE, &m_ntIcon);\n    if (theApp.m_cfg_data.m_show_task_bar_wnd)\n        OpenTaskBarWnd();\n}\n\nvoid CTrafficMonitorDlg::ShowNotifyTip(const wchar_t* title, const wchar_t* message)\n{\n    //要显示通知区提示，必须先将通知区图标显示出来\n    if (!theApp.m_general_data.show_notify_icon)\n    {\n        //添加通知栏图标\n        AddNotifyIcon();\n        theApp.m_general_data.show_notify_icon = true;\n    }\n    //显示通知提示\n    m_ntIcon.uFlags |= NIF_INFO;\n    //wcscpy_s(m_ntIcon.szInfo, message ? message : _T(\"\"));\n    //wcscpy_s(m_ntIcon.szInfoTitle, title ? title : _T(\"\"));\n    CCommon::WStringCopy(m_ntIcon.szInfo, 256, message);\n    CCommon::WStringCopy(m_ntIcon.szInfoTitle, 64, title);\n    ::Shell_NotifyIcon(NIM_MODIFY, &m_ntIcon);\n    m_ntIcon.uFlags &= ~NIF_INFO;\n}\n\nvoid CTrafficMonitorDlg::UpdateNotifyIconTip()\n{\n    CString strTip;         //鼠标指向图标时显示的提示\n#ifdef _DEBUG\n    strTip = CCommon::LoadText(IDS_TRAFFICMONITOR, _T(\" (Debug)\"));\n#else\n    strTip = CCommon::LoadText(IDS_TRAFFICMONITOR);\n#endif\n\n    CString in_speed = CCommon::DataSizeToString(theApp.m_in_speed);\n    CString out_speed = CCommon::DataSizeToString(theApp.m_out_speed);\n\n    strTip += CCommon::StringFormat(_T(\"\\r\\n<%1%>: <%2%>/s\"), { CCommon::LoadText(IDS_UPLOAD), out_speed });\n    strTip += CCommon::StringFormat(_T(\"\\r\\n<%1%>: <%2%>/s\"), { CCommon::LoadText(IDS_DOWNLOAD), in_speed });\n    strTip += CCommon::StringFormat(_T(\"\\r\\nCPU: <%1%> %\"), { theApp.m_cpu_usage });\n    strTip += CCommon::StringFormat(_T(\"\\r\\n<%1%>: <%2%> %\"), { CCommon::LoadText(IDS_MEMORY), theApp.m_memory_usage });\n    if (IsTemperatureNeeded())\n    {\n        if (theApp.m_general_data.IsHardwareEnable(HI_GPU) && theApp.m_gpu_usage >= 0)\n            strTip += CCommon::StringFormat(_T(\"\\r\\n<%1%>: <%2%> %\"), { CCommon::LoadText(IDS_GPU_USAGE), theApp.m_gpu_usage });\n        if (theApp.m_general_data.IsHardwareEnable(HI_CPU) && theApp.m_cpu_temperature > 0)\n            strTip += CCommon::StringFormat(_T(\"\\r\\n<%1%>: <%2%> °C\"), { CCommon::LoadText(IDS_CPU_TEMPERATURE), static_cast<int>(theApp.m_cpu_temperature) });\n        if (theApp.m_general_data.IsHardwareEnable(HI_GPU) && theApp.m_gpu_temperature > 0)\n            strTip += CCommon::StringFormat(_T(\"\\r\\n<%1%>: <%2%> °C\"), { CCommon::LoadText(IDS_GPU_TEMPERATURE), static_cast<int>(theApp.m_gpu_temperature) });\n        if (theApp.m_general_data.IsHardwareEnable(HI_HDD) && theApp.m_hdd_temperature > 0)\n            strTip += CCommon::StringFormat(_T(\"\\r\\n<%1%>: <%2%> °C\"), { CCommon::LoadText(IDS_HDD_TEMPERATURE), static_cast<int>(theApp.m_hdd_temperature) });\n        if (theApp.m_general_data.IsHardwareEnable(HI_MBD) && theApp.m_main_board_temperature > 0)\n            strTip += CCommon::StringFormat(_T(\"\\r\\n<%1%>: <%2%> °C\"), { CCommon::LoadText(IDS_MAINBOARD_TEMPERATURE), static_cast<int>(theApp.m_main_board_temperature) });\n        if (theApp.m_general_data.IsHardwareEnable(HI_HDD) && theApp.m_hdd_usage >= 0)\n            strTip += CCommon::StringFormat(_T(\"\\r\\n<%1%>: <%2%> %\"), { CCommon::LoadText(IDS_HDD_USAGE), theApp.m_hdd_usage });\n    }\n\n    CCommon::WStringCopy(m_ntIcon.szTip, 128, strTip);\n    ::Shell_NotifyIcon(NIM_MODIFY, &m_ntIcon);\n\n}\n\nvoid CTrafficMonitorDlg::SaveHistoryTraffic()\n{\n    m_history_traffic.Save();\n}\n\nvoid CTrafficMonitorDlg::LoadHistoryTraffic()\n{\n    m_history_traffic.Load();\n    CHistoryTrafficFile backup_file(theApp.m_history_traffic_path + L\".bak\");\n    backup_file.LoadSize();     //读取备份文件中流量记录的数量\n    if (backup_file.Size() > m_history_traffic.Size())      //如果备份文件中流量记录的数量大于当前的数量，则从备份文件中恢复\n    {\n        backup_file.Load();\n        size_t size_before = m_history_traffic.Size();\n        m_history_traffic.Merge(backup_file, true);\n        CString log_info = CCommon::LoadTextFormat(IDS_HISTORY_TRAFFIC_LOST_ERROR_LOG, { size_before, backup_file.Size() });\n        CCommon::WriteLog(log_info, theApp.m_log_path.c_str());\n    }\n\n    theApp.m_today_up_traffic = m_history_traffic.GetTodayUpTraffic();\n    theApp.m_today_down_traffic = m_history_traffic.GetTodayDownTraffic();\n}\n\nvoid CTrafficMonitorDlg::BackupHistoryTrafficFile()\n{\n    CHistoryTrafficFile backup_file(theApp.m_history_traffic_path + L\".bak\");\n    CHistoryTrafficFile latest_file(theApp.m_history_traffic_path);\n    backup_file.LoadSize();\n    latest_file.LoadSize();\n    if (backup_file.Size() < latest_file.Size())\n    {\n        CopyFile(latest_file.GetFilePath().c_str(), backup_file.GetFilePath().c_str(), FALSE);\n    }\n}\n\nvoid CTrafficMonitorDlg::_OnOptions(int tab)\n{\n    COptionsDlg optionsDlg(tab, this);\n\n    //将选项设置数据传递给选项设置对话框\n    if (COptionsDlg::GetUniqueHandel(OPTION_DLG_NAME) == NULL)     //确保此时选项设置对话框已经关闭\n    {\n        optionsDlg.m_tab1_dlg.m_data = theApp.m_main_wnd_data;\n        optionsDlg.m_tab2_dlg.m_data = theApp.m_taskbar_data;\n        optionsDlg.m_tab3_dlg.m_data = theApp.m_general_data;\n        optionsDlg.m_tab1_dlg.m_text_disable = m_skin.GetLayoutInfo().no_label;\n    }\n\n    if (optionsDlg.DoModal() == IDOK)\n    {\n        ApplySettings(optionsDlg);\n    }\n}\n\nvoid CTrafficMonitorDlg::ApplySettings(COptionsDlg& optionsDlg)\n{\n    bool is_hardware_monitor_item_changed = (optionsDlg.m_tab3_dlg.m_data.hardware_monitor_item != theApp.m_general_data.hardware_monitor_item);\n    bool is_always_on_top_changed = (optionsDlg.m_tab1_dlg.m_data.m_always_on_top != theApp.m_main_wnd_data.m_always_on_top);\n    bool is_mouse_penerate_changed = (optionsDlg.m_tab1_dlg.m_data.m_mouse_penetrate != theApp.m_main_wnd_data.m_mouse_penetrate);\n    bool is_alow_out_of_border_changed = (optionsDlg.m_tab1_dlg.m_data.m_alow_out_of_border != theApp.m_main_wnd_data.m_alow_out_of_border);\n    bool is_show_notify_icon_changed = (optionsDlg.m_tab3_dlg.m_data.show_notify_icon != theApp.m_general_data.show_notify_icon);\n    bool is_connections_hide_changed = (optionsDlg.m_tab3_dlg.m_data.connections_hide.data() != theApp.m_general_data.connections_hide.data());\n\n    theApp.m_main_wnd_data = optionsDlg.m_tab1_dlg.m_data;\n    theApp.m_taskbar_data = optionsDlg.m_tab2_dlg.m_data;\n    theApp.m_general_data = optionsDlg.m_tab3_dlg.m_data;\n    theApp.SendSettingsToPlugin();\n\n    CGeneralSettingsDlg::CheckTaskbarDisplayItem();\n\n    SetTextFont();\n\n    //CTaskBarDlg::SaveConfig();\n    if (IsTaskbarWndValid())\n    {\n        m_tBarDlg->ApplySettings();\n        //如果更改了任务栏窗口字体或显示的文本，则任务栏窗口可能要变化，于是关闭再打开任务栏窗口\n        CloseTaskBarWnd();\n        OpenTaskBarWnd();\n    }\n\n    if (optionsDlg.m_tab3_dlg.IsAutoRunModified())\n    {\n        if (!theApp.SetAutoRun(theApp.m_general_data.auto_run))\n            MessageBox(CCommon::LoadText(IDS_SET_AUTO_RUN_FAILED_WARNING), NULL, MB_ICONWARNING | MB_OK);\n    }\n\n    if (optionsDlg.m_tab3_dlg.IsShowAllInterfaceModified() || is_connections_hide_changed)\n        IniConnection();\n\n    if (optionsDlg.m_tab3_dlg.IsMonitorTimeSpanModified())      //如果监控时间间隔改变了，则重设定时器\n    {\n        KillTimer(MONITOR_TIMER);\n        SetTimer(MONITOR_TIMER, theApp.m_general_data.monitor_time_span, NULL);\n        //m_timer.KillTimer();\n        //m_timer.CreateTimer((DWORD_PTR)this, theApp.m_general_data.monitor_time_span, MonitorThreadCallback);\n    }\n\n    //设置获取CPU利用率的方式\n    m_cpu_usage.SetUseCPUTimes(theApp.m_general_data.m_get_cpu_usage_by_cpu_times);\n\n#ifndef WITHOUT_TEMPERATURE\n    if (is_hardware_monitor_item_changed)\n    {\n        //如果关闭了硬件监控，则析构硬件监控类\n        if (theApp.m_general_data.hardware_monitor_item == 0)\n        {\n            CSingleLock sync(&theApp.m_minitor_lib_critical, TRUE);\n            theApp.m_pMonitor.reset();\n        }\n        else if (theApp.m_pMonitor != nullptr)\n        {\n            theApp.UpdateOpenHardwareMonitorEnableState();\n        }\n        else if (IsTemperatureNeeded())\n        {\n            theApp.InitOpenHardwareLibInThread();\n        }\n        //更新任务栏窗口右键菜单\n        //theApp.UpdateTaskbarWndMenu();\n    }\n#endif\n\n    if (is_always_on_top_changed)\n    {\n        SetAlwaysOnTop();\n    }\n\n    if (is_mouse_penerate_changed)\n    {\n        SetMousePenetrate();\n        if (!theApp.m_general_data.show_notify_icon && theApp.IsForceShowNotifyIcon())   //鼠标穿透时，如果通知图标没有显示，则将它显示出来，否则无法呼出右键菜单\n        {\n            //添加通知栏图标\n            AddNotifyIcon();\n            theApp.m_general_data.show_notify_icon = true;\n        }\n    }\n\n    if (is_alow_out_of_border_changed)\n    {\n        CheckWindowPos();\n    }\n\n    if (is_show_notify_icon_changed)\n    {\n        if (theApp.IsForceShowNotifyIcon())\n            theApp.m_general_data.show_notify_icon = true;\n        if (theApp.m_general_data.show_notify_icon)\n            AddNotifyIcon();\n        else\n            DeleteNotifyIcon();\n    }\n\n    theApp.SaveConfig();\n    theApp.SaveGlobalConfig();\n}\n\nvoid CTrafficMonitorDlg::SetItemPosition()\n{\n    if (theApp.m_cfg_data.m_show_more_info)\n    {\n        SetWindowPos(nullptr, 0, 0, m_skin.GetLayoutInfo().layout_l.width, m_skin.GetLayoutInfo().layout_l.height, SWP_NOMOVE | SWP_NOZORDER);\n    }\n    else\n    {\n        SetWindowPos(nullptr, 0, 0, m_skin.GetLayoutInfo().layout_s.width, m_skin.GetLayoutInfo().layout_s.height, SWP_NOMOVE | SWP_NOZORDER);\n    }\n}\n\nvoid CTrafficMonitorDlg::LoadSkinLayout()\n{\n    wstring skin_cfg_path{ theApp.m_skin_path + m_skins[m_skin_selected] + L\"\\\\skin.xml\" };\n    if (!CCommon::FileExist(skin_cfg_path.c_str()))\n        skin_cfg_path = theApp.m_skin_path + m_skins[m_skin_selected] + L\"\\\\skin.ini\";\n    m_skin.Load(skin_cfg_path);\n    if (m_skin.GetLayoutInfo().no_label)        //如果皮肤布局不显示文本，则不允许交换上传和下载的位置，因为上传和下载的位置已经固定在皮肤中了\n        theApp.m_main_wnd_data.swap_up_down = false;\n}\n\nvoid CTrafficMonitorDlg::LoadBackGroundImage()\n{\n    CImage img_tmp;\n    CSize image_size;\n    if (theApp.m_cfg_data.m_show_more_info)\n    {\n        image_size.SetSize(m_skin.GetLayoutInfo().layout_l.width, m_skin.GetLayoutInfo().layout_l.height);\n    }\n    else\n    {\n        image_size.SetSize(m_skin.GetLayoutInfo().layout_s.width, m_skin.GetLayoutInfo().layout_s.height);\n    }\n\n    //创建窗口区域\n    CImage img_mask;\n    //载入掩码图片\n    if (theApp.m_cfg_data.m_show_more_info)\n        img_tmp.Load((theApp.m_skin_path + m_skins[m_skin_selected] + BACKGROUND_MASK_L).c_str());\n    else\n        img_tmp.Load((theApp.m_skin_path + m_skins[m_skin_selected] + BACKGROUND_MASK_S).c_str());\n    CRgn wnd_rgn;\n    if (!img_tmp.IsNull())\n    {\n        CDrawCommon::BitmapStretch(&img_tmp, &img_mask, image_size);        //拉伸掩码图片\n        CBitmap bitmap;\n        bitmap.Attach(img_mask);\n        CDrawCommon::GetRegionFromImage(wnd_rgn, bitmap, 128);      //从掩码图片获得窗口的区域\n        bitmap.Detach();\n    }\n    else\n    {\n        wnd_rgn.CreateRectRgnIndirect(CRect(CPoint(0, 0), image_size));     //载入掩码图片失败，则使用窗口大小作为窗口区域\n    }\n    //避免获取到的窗口区域为空\n    CRgn empty_rgn;\n    empty_rgn.CreateRectRgnIndirect(CRect{});   //创建一个空的CRgn对象\n    if (wnd_rgn.EqualRgn(&empty_rgn))\n        wnd_rgn.SetRectRgn(CRect(CPoint(0, 0), image_size));    //如果获取到的窗口区域为空，则使用窗口大小作为窗口区域\n    SetWindowRgn(wnd_rgn, TRUE);        //设置窗口区域\n    wnd_rgn.DeleteObject();\n    empty_rgn.DeleteObject();\n}\n\nvoid CTrafficMonitorDlg::SetTextFont()\n{\n    if (m_font.m_hObject)   //如果m_font已经关联了一个字体资源对象，则释放它\n        m_font.DeleteObject();\n    theApp.m_main_wnd_data.font.Create(m_font, theApp.GetDpi());\n}\n\nbool CTrafficMonitorDlg::IsTaskbarWndValid() const\n{\n    return m_tBarDlg != nullptr && ::IsWindow(m_tBarDlg->GetSafeHwnd());\n}\n\nvoid CTrafficMonitorDlg::TaskbarShowHideItem(DisplayItem type)\n{\n    if (IsTaskbarWndValid())\n    {\n        bool show = (theApp.m_taskbar_data.m_tbar_display_item & type);\n        if (show)\n        {\n            theApp.m_taskbar_data.m_tbar_display_item &= ~type;\n        }\n        else\n        {\n            theApp.m_taskbar_data.m_tbar_display_item |= type;\n        }\n        CloseTaskBarWnd();\n        OpenTaskBarWnd();\n    }\n}\n\nvoid CTrafficMonitorDlg::CheckClickedItem(CPoint point)\n{\n    const CSkinFile::Layout& skin_layout{ theApp.m_cfg_data.m_show_more_info ? m_skin.GetLayoutInfo().layout_l : m_skin.GetLayoutInfo().layout_s }; //当前的皮肤布局\n    for (const auto& layout_item : skin_layout.layout_items)\n    {\n        CRect rect(CPoint(layout_item.second.x, layout_item.second.y), CSize(layout_item.second.width, m_skin.GetLayoutInfo().text_height));\n        if (rect.PtInRect(point))\n        {\n            m_clicked_item = layout_item.first;\n            break;\n        }\n    }\n}\n\nbool CTrafficMonitorDlg::IsTemperatureNeeded() const\n{\n    //判断是否需要从OpenHardwareMonitor获取信息。\n    ////只有主窗口和任务栏窗口中CPU温度、显卡利用率、显卡温度、硬盘温度和主板温度中至少有一个要显示，才返回true\n    //bool needed = false;\n    //if (theApp.m_cfg_data.m_show_task_bar_wnd && IsTaskbarWndValid())\n    //{\n    //    needed |= m_tBarDlg->IsShowCpuTemperature();\n    //    needed |= m_tBarDlg->IsShowGpu();\n    //    needed |= m_tBarDlg->IsShowGpuTemperature();\n    //    needed |= m_tBarDlg->IsShowHddTemperature();\n    //    needed |= m_tBarDlg->IsShowMainboardTemperature();\n    //    needed |= (::IsWindow(m_tBarDlg->m_tool_tips.GetSafeHwnd()) && m_tBarDlg->m_tool_tips.IsWindowVisible());\n    //}\n\n    //if (!theApp.m_cfg_data.m_hide_main_window)\n    //{\n    //    const CSkinFile::Layout& skin_layout{ theApp.m_cfg_data.m_show_more_info ? m_skin.GetLayoutInfo().layout_l : m_skin.GetLayoutInfo().layout_s }; //当前的皮肤布局\n    //    needed |= skin_layout.GetItem(TDI_CPU_TEMP).show;\n    //    needed |= skin_layout.GetItem(TDI_GPU_USAGE).show;\n    //    needed |= skin_layout.GetItem(TDI_GPU_TEMP).show;\n    //    needed |= skin_layout.GetItem(TDI_HDD_TEMP).show;\n    //    needed |= skin_layout.GetItem(TDI_MAIN_BOARD_TEMP).show;\n    //    needed |= (::IsWindow(m_tool_tips.GetSafeHwnd()) && m_tool_tips.IsWindowVisible());\n    //}\n    //return needed;\n\n    return theApp.m_general_data.IsHardwareEnable(HI_CPU) || theApp.m_general_data.IsHardwareEnable(HI_GPU)\n        || theApp.m_general_data.IsHardwareEnable(HI_HDD) || theApp.m_general_data.IsHardwareEnable(HI_MBD);\n}\n\n// CTrafficMonitorDlg 消息处理程序\n\nBOOL CTrafficMonitorDlg::OnInitDialog()\n{\n    CDialog::OnInitDialog();\n\n    // 设置此对话框的图标。  当应用程序主窗口不是对话框时，框架将自动\n    //  执行此操作\n    SetIcon(m_hIcon, TRUE);         // 设置大图标\n    SetIcon(m_hIcon, FALSE);        // 设置小图标\n\n    // TODO: 在此添加额外的初始化代码\n    SetWindowText(APP_NAME);\n    //设置隐藏任务栏图标\n    ModifyStyleEx(WS_EX_APPWINDOW, WS_EX_TOOLWINDOW);\n\n    theApp.DPIFromWindow(this);\n    //获取屏幕大小\n    GetScreenSize();\n    m_last_screen_rects = m_screen_rects;\n    //::SystemParametersInfo(SPI_GETWORKAREA, 0, &m_screen_rect, 0);   // 获得工作区大小\n\n    //初始化菜单\n    theApp.InitMenuResourse();\n    //theApp.UpdateTaskbarWndMenu();\n\n    //设置窗口透明度\n    SetTransparency();\n\n    IniConnection();    //初始化连接\n\n    //如果启动时设置了鼠标穿透或隐藏主窗口，并且没有显示任务栏窗口，则显示通知区图标\n    if ((theApp.m_main_wnd_data.m_mouse_penetrate || theApp.m_cfg_data.m_hide_main_window) && !theApp.m_cfg_data.m_show_task_bar_wnd)\n        theApp.m_general_data.show_notify_icon = true;\n\n    //载入通知区图标\n    theApp.m_notify_icons[0] = (HICON)LoadImage(AfxGetInstanceHandle(), MAKEINTRESOURCE(IDI_NOFITY_ICON), IMAGE_ICON, theApp.DPI(16), theApp.DPI(16), LR_DEFAULTCOLOR | LR_CREATEDIBSECTION);\n    theApp.m_notify_icons[1] = (HICON)LoadImage(AfxGetInstanceHandle(), MAKEINTRESOURCE(IDI_NOFITY_ICON2), IMAGE_ICON, theApp.DPI(16), theApp.DPI(16), LR_DEFAULTCOLOR | LR_CREATEDIBSECTION);\n    theApp.m_notify_icons[2] = (HICON)LoadImage(AfxGetInstanceHandle(), MAKEINTRESOURCE(IDI_NOFITY_ICON3), IMAGE_ICON, theApp.DPI(16), theApp.DPI(16), LR_DEFAULTCOLOR | LR_CREATEDIBSECTION);\n    theApp.m_notify_icons[3] = (HICON)LoadImage(AfxGetInstanceHandle(), MAKEINTRESOURCE(IDR_MAINFRAME), IMAGE_ICON, theApp.DPI(16), theApp.DPI(16), LR_DEFAULTCOLOR | LR_CREATEDIBSECTION);\n    theApp.m_notify_icons[4] = (HICON)LoadImage(AfxGetInstanceHandle(), MAKEINTRESOURCE(IDI_NOFITY_ICON4), IMAGE_ICON, theApp.DPI(16), theApp.DPI(16), LR_DEFAULTCOLOR | LR_CREATEDIBSECTION);\n    theApp.m_notify_icons[5] = (HICON)LoadImage(AfxGetInstanceHandle(), MAKEINTRESOURCE(IDI_NOTIFY_ICON5), IMAGE_ICON, theApp.DPI(16), theApp.DPI(16), LR_DEFAULTCOLOR | LR_CREATEDIBSECTION);\n\n    //设置通知区域图标\n    m_ntIcon.cbSize = sizeof(NOTIFYICONDATA);   //该结构体变量的大小\n    if (theApp.m_cfg_data.m_notify_icon_selected < 0 || theApp.m_cfg_data.m_notify_icon_selected >= MAX_NOTIFY_ICON)\n        theApp.m_cfg_data.m_notify_icon_selected = 0;\n    m_ntIcon.hIcon = theApp.m_notify_icons[theApp.m_cfg_data.m_notify_icon_selected];       //设置图标\n    m_ntIcon.hWnd = this->m_hWnd;               //接收托盘图标通知消息的窗口句柄\n    CString atip;           //鼠标指向图标时显示的提示\n#ifdef _DEBUG\n    atip = CCommon::LoadText(IDS_TRAFFICMONITOR, _T(\" (Debug)\"));\n#else\n    atip = CCommon::LoadText(IDS_TRAFFICMONITOR);\n#endif\n    //wcscpy_s(m_ntIcon.szTip, 128, strTip);\n    CCommon::WStringCopy(m_ntIcon.szTip, 128, atip.GetString());\n    m_ntIcon.uCallbackMessage = MY_WM_NOTIFYICON;   //应用程序定义的消息ID号\n    m_ntIcon.uFlags = NIF_MESSAGE | NIF_ICON | NIF_TIP; //图标的属性：设置成员uCallbackMessage、hIcon、szTip有效\n    if (theApp.m_general_data.show_notify_icon)\n        ::Shell_NotifyIcon(NIM_ADD, &m_ntIcon); //在系统通知区域增加这个图标\n\n    //载入流量历史记录\n    LoadHistoryTraffic();\n\n    //设置1000毫秒触发的定时器\n    SetTimer(MAIN_TIMER, 1000, NULL);\n\n    SetTimer(MONITOR_TIMER, theApp.m_general_data.monitor_time_span, NULL);\n    //m_timer.CreateTimer((DWORD_PTR)this, theApp.m_general_data.monitor_time_span, MonitorThreadCallback);\n\n\n    //初始化皮肤\n    CCommon::GetFiles((theApp.m_skin_path + L\"\\\\*\").c_str(), [&](const wstring& file_name)\n        {\n            wstring file_name1 = L'\\\\' + file_name;\n            if (CCommon::IsFolder(theApp.m_skin_path + file_name1))\n                m_skins.push_back(file_name1);\n        });\n    if (m_skins.empty())\n        m_skins.push_back(L\"\");\n    m_skin_selected = 0;\n    for (unsigned int i{}; i < m_skins.size(); i++)\n    {\n        if (m_skins[i] == theApp.m_cfg_data.m_skin_name)\n            m_skin_selected = i;\n    }\n\n    //根据当前选择的皮肤获取布局数据\n    LoadSkinLayout();\n\n    //初始化窗口位置\n    SetItemPosition();\n    if (theApp.m_cfg_data.m_position_x != -1 && theApp.m_cfg_data.m_position_y != -1)\n        SetWindowPos(nullptr, theApp.m_cfg_data.m_position_x, theApp.m_cfg_data.m_position_y, 0, 0, SWP_NOZORDER | SWP_NOSIZE);\n    CheckWindowPos();\n\n    //载入背景图片\n    LoadBackGroundImage();\n\n    //设置字体\n    SetTextFont();\n\n    //获取启动时的时间\n    GetLocalTime(&m_start_time);\n\n    //初始化鼠标提示\n    m_tool_tips.Create(this, TTS_ALWAYSTIP);\n    m_tool_tips.SetMaxTipWidth(600);\n    m_tool_tips.AddTool(this, _T(\"\"));\n\n    //设置获取CPU利用率的方式\n    m_cpu_usage.SetUseCPUTimes(theApp.m_general_data.m_get_cpu_usage_by_cpu_times);\n\n    //如果程序启动时设置了隐藏主窗口，或窗口的位置在左上角，则先将其不透明度设为0\n    if (theApp.m_cfg_data.m_hide_main_window || (theApp.m_cfg_data.m_position_x == 0 && theApp.m_cfg_data.m_position_y == 0))\n        SetTransparency(0);\n\n    SetTimer(TASKBAR_TIMER, 100, NULL);\n\n    return TRUE;  // 除非将焦点设置到控件，否则返回 TRUE\n}\n\n\n//当用户拖动最小化窗口时系统调用此函数取得光标\n//显示。\nHCURSOR CTrafficMonitorDlg::OnQueryDragIcon()\n{\n    return static_cast<HCURSOR>(m_hIcon);\n}\n\n//计算指定秒数的时间内Monitor定时器会触发的次数\nstatic int GetMonitorTimerCount(int second)\n{\n    return second * 1000 / theApp.m_general_data.monitor_time_span;\n}\n\n\nUINT CTrafficMonitorDlg::MonitorThreadCallback(LPVOID dwUser)\n{\n    CTrafficMonitorDlg* pThis = (CTrafficMonitorDlg*)dwUser;\n    CFlagLocker flag_locker(pThis->m_is_monitor_thread_runing);\n\n    //获取网络连接速度\n    int rtn{};\n    auto getLfTable = [&]()\n    {\n        __try\n        {\n            rtn = GetIfTable(pThis->m_pIfTable, &pThis->m_dwSize, FALSE);\n        }\n        __except (EXCEPTION_EXECUTE_HANDLER)\n        {\n            free(pThis->m_pIfTable);\n            pThis->m_dwSize = sizeof(MIB_IFTABLE);\n            pThis->m_pIfTable = (MIB_IFTABLE*)malloc(pThis->m_dwSize);\n            rtn = GetIfTable(pThis->m_pIfTable, &pThis->m_dwSize, FALSE);\n            if (rtn == ERROR_INSUFFICIENT_BUFFER)\t//如果函数返回值为ERROR_INSUFFICIENT_BUFFER，说明m_pIfTable的大小不够\n            {\n                free(pThis->m_pIfTable);\n                pThis->m_pIfTable = (MIB_IFTABLE*)malloc(pThis->m_dwSize);\t//用新的大小重新开辟一块内存\n            }\n            GetIfTable(pThis->m_pIfTable, &pThis->m_dwSize, FALSE);\n        }\n    };\n\n    getLfTable();\n\n    if (!theApp.m_cfg_data.m_select_all)        //获取当前选中连接的网速\n    {\n        auto table = pThis->GetConnectIfTable(pThis->m_connection_selected);\n        pThis->m_in_bytes = table.dwInOctets;\n        pThis->m_out_bytes = table.dwOutOctets;\n    }\n    else        //获取全部连接的网速\n    {\n        pThis->m_in_bytes = 0;\n        pThis->m_out_bytes = 0;\n        for (size_t i{}; i < pThis->m_connections.size(); i++)\n        {\n            auto table = pThis->GetConnectIfTable(i);\n            //if (i > 0 && m_pIfTable->table[m_connections[i].index].dwInOctets == m_pIfTable->table[m_connections[i - 1].index].dwInOctets\n            //  && m_pIfTable->table[m_connections[i].index].dwOutOctets == m_pIfTable->table[m_connections[i - 1].index].dwOutOctets)\n            //  continue;       //连接列表中可能会有相同的连接，统计所有连接的网速时，忽略掉已发送和已接收字节数完全相同的连接\n            pThis->m_in_bytes += table.dwInOctets;\n            pThis->m_out_bytes += table.dwOutOctets;\n        }\n    }\n\n    unsigned __int64 cur_in_speed{}, cur_out_speed{};       //本次监控时间间隔内的上传和下载速度\n\n    //如果发送和接收的字节数为0或上次发送和接收的字节数为0或当前连接已改变时，网速无效\n    if ((pThis->m_in_bytes == 0 && pThis->m_out_bytes == 0) || (pThis->m_last_in_bytes == 0 && pThis->m_last_out_bytes == 0) || pThis->m_connection_change_flag\n        || pThis->m_last_in_bytes > pThis->m_in_bytes || pThis->m_last_out_bytes > pThis->m_out_bytes)\n    {\n        cur_in_speed = 0;\n        cur_out_speed = 0;\n    }\n    else\n    {\n        cur_in_speed = pThis->m_in_bytes - pThis->m_last_in_bytes;\n        cur_out_speed = pThis->m_out_bytes - pThis->m_last_out_bytes;\n    }\n    ////如果大于1GB/s，说明可能产生了异常，网速无效\n    //if (cur_in_speed > 1073741824)\n    //  cur_in_speed = 0;\n    //if (cur_out_speed > 1073741824)\n    //  cur_out_speed = 0;\n\n    //将当前监控时间间隔的流量转换成每秒时间间隔内的流量\n    theApp.m_in_speed = static_cast<unsigned __int64>(cur_in_speed * 1000 / theApp.m_general_data.monitor_time_span);\n    theApp.m_out_speed = static_cast<unsigned __int64>(cur_out_speed * 1000 / theApp.m_general_data.monitor_time_span);\n\n    pThis->m_connection_change_flag = false;    //清除连接发生变化的标志\n\n    pThis->m_last_in_bytes = pThis->m_in_bytes;\n    pThis->m_last_out_bytes = pThis->m_out_bytes;\n\n    //处于自动选择状态时，如果连续30秒没有网速，则可能自动选择的网络不对，此时执行一次自动选择\n    if (theApp.m_cfg_data.m_auto_select)\n    {\n        if (cur_in_speed == 0 && cur_out_speed == 0)\n            pThis->m_zero_speed_cnt++;\n        else\n            pThis->m_zero_speed_cnt = 0;\n        if (pThis->m_zero_speed_cnt >= GetMonitorTimerCount(30))\n        {\n            pThis->AutoSelect();\n            pThis->m_zero_speed_cnt = 0;\n        }\n    }\n\n    //检测当前日期是否改变，如果已改变，就向历史流量列表插入一个新的日期\n    SYSTEMTIME current_time;\n    GetLocalTime(&current_time);\n    if (pThis->m_history_traffic.GetTraffics()[0].day != current_time.wDay)\n    {\n        HistoryTraffic traffic;\n        traffic.year = current_time.wYear;\n        traffic.month = current_time.wMonth;\n        traffic.day = current_time.wDay;\n        traffic.mixed = false;\n        pThis->m_history_traffic.GetTraffics().push_front(traffic);\n        theApp.m_today_up_traffic = 0;\n        theApp.m_today_down_traffic = 0;\n    }\n\n    //统计今天已使用的流量\n    theApp.m_today_up_traffic += cur_out_speed;\n    theApp.m_today_down_traffic += cur_in_speed;\n    pThis->m_history_traffic.GetTraffics()[0].up_kBytes = theApp.m_today_up_traffic / 1024u;\n    pThis->m_history_traffic.GetTraffics()[0].down_kBytes = theApp.m_today_down_traffic / 1024u;\n    //每隔30秒保存一次流量历史记录\n    if (pThis->m_monitor_time_cnt % GetMonitorTimerCount(30) == GetMonitorTimerCount(30) - 1)\n    {\n        static unsigned __int64 last_today_kbytes;\n        if (pThis->m_history_traffic.GetTraffics()[0].kBytes() - last_today_kbytes >= 100u) //只有当流量变化超过100KB时才保存历史流量记录，防止磁盘写入过于频繁\n        {\n            pThis->SaveHistoryTraffic();\n            last_today_kbytes = pThis->m_history_traffic.GetTraffics()[0].kBytes();\n        }\n    }\n\n    if (rtn == ERROR_INSUFFICIENT_BUFFER)\n    {\n        pThis->IniConnection();\n        CString info;\n        info.LoadString(IDS_INSUFFICIENT_BUFFER);\n        info.Replace(_T(\"<%cnt%>\"), CCommon::IntToString(pThis->m_restart_cnt));\n        CCommon::WriteLog(info, theApp.m_log_path.c_str());\n    }\n\n\n    if (pThis->m_monitor_time_cnt % GetMonitorTimerCount(3) == GetMonitorTimerCount(3) - 1)\n    {\n        //重新获取当前连接数量\n        static DWORD last_interface_num = -1;\n        DWORD interface_num;\n        GetNumberOfInterfaces(&interface_num);\n        if (last_interface_num != -1 && interface_num != last_interface_num)    //如果连接数发生变化，则重新初始化连接\n        {\n            if (theApp.m_debug_log)\n            {\n                CString info;\n                info.LoadString(IDS_CONNECTION_NUM_CHANGED);\n                info.Replace(_T(\"<%before%>\"), CCommon::IntToString(last_interface_num));\n                info.Replace(_T(\"<%after%>\"), CCommon::IntToString(interface_num));\n                info.Replace(_T(\"<%cnt%>\"), CCommon::IntToString(pThis->m_restart_cnt + 1));\n                CCommon::WriteLog(info, theApp.m_log_path.c_str());\n            }\n            pThis->IniConnection();\n            last_interface_num = interface_num;\n        }\n\n        string descr;\n        descr = (const char*)pThis->GetConnectIfTable(pThis->m_connection_selected).bDescr;\n        if (descr != theApp.m_cfg_data.m_connection_name)\n        {\n            //写入额外的调试信息\n            if (theApp.m_debug_log)\n            {\n                CString log_str;\n                log_str = _T(\"连接名称不匹配：\\r\\n\");\n                log_str += _T(\"IfTable description: \");\n                log_str += descr.c_str();\n                log_str += _T(\"\\r\\nm_connection_name: \");\n                log_str += theApp.m_cfg_data.m_connection_name.c_str();\n                CCommon::WriteLog(log_str, (theApp.m_config_dir + L\".\\\\connections.log\").c_str());\n            }\n\n            pThis->IniConnection();\n            CString info;\n            info.LoadString(IDS_CONNECTION_NOT_MATCH);\n            info.Replace(_T(\"<%cnt%>\"), CCommon::IntToString(pThis->m_restart_cnt));\n            CCommon::WriteLog(info, theApp.m_log_path.c_str());\n        }\n    }\n\n    ////只有主窗口和任务栏窗口至少有一个显示时才执行下面的处理\n    //if (!theApp.m_cfg_data.m_hide_main_window || theApp.m_cfg_data.m_show_task_bar_wnd)\n    //{\n    //获取CPU使用率\n    theApp.m_cpu_usage = pThis->m_cpu_usage.GetCPUUsage();\n\n    //获取内存利用率\n    MEMORYSTATUSEX statex;\n    statex.dwLength = sizeof(statex);\n    GlobalMemoryStatusEx(&statex);\n    theApp.m_memory_usage = statex.dwMemoryLoad;\n    theApp.m_used_memory = static_cast<int>((statex.ullTotalPhys - statex.ullAvailPhys) / 1024);\n    theApp.m_total_memory = static_cast<int>(statex.ullTotalPhys / 1024);\n\n#ifndef WITHOUT_TEMPERATURE\n    //获取温度\n    if (pThis->IsTemperatureNeeded() && theApp.m_pMonitor != nullptr)\n    {\n        CSingleLock sync(&theApp.m_minitor_lib_critical, TRUE);\n\n        auto getHardwareInfo = []()\n        {\n            __try\n            {\n                theApp.m_pMonitor->GetHardwareInfo();\n            }\n            __except (EXCEPTION_EXECUTE_HANDLER)\n            {\n                AfxMessageBox(IDS_HARDWARE_INFO_ACQUIRE_FAILED_ERROR, MB_ICONERROR | MB_OK);\n            }\n        };\n\n        getHardwareInfo();\n        auto monitor_error_message{ OpenHardwareMonitorApi::GetErrorMessage() };\n        if (!monitor_error_message.empty())\n        {\n            AfxMessageBox(monitor_error_message.c_str(), MB_ICONERROR | MB_OK);\n        }\n        //theApp.m_cpu_temperature = theApp.m_pMonitor->CpuTemperature();\n        theApp.m_gpu_temperature = theApp.m_pMonitor->GpuTemperature();\n        //theApp.m_hdd_temperature = theApp.m_pMonitor->HDDTemperature();\n        theApp.m_main_board_temperature = theApp.m_pMonitor->MainboardTemperature();\n        theApp.m_gpu_usage = theApp.m_pMonitor->GpuUsage();\n        theApp.m_cpu_freq = theApp.m_pMonitor->CpuFreq();\n        //获取CPU温度\n        if (!theApp.m_pMonitor->AllCpuTemperature().empty())\n        {\n            if (theApp.m_general_data.cpu_core_name == CCommon::LoadText(IDS_AVREAGE_TEMPERATURE).GetString())  //如果选择了平均温度\n            {\n                theApp.m_cpu_temperature = theApp.m_pMonitor->CpuTemperature();\n            }\n            else\n            {\n                auto iter = theApp.m_pMonitor->AllCpuTemperature().find(theApp.m_general_data.cpu_core_name);\n                if (iter == theApp.m_pMonitor->AllCpuTemperature().end())\n                {\n                    iter = theApp.m_pMonitor->AllCpuTemperature().begin();\n                    theApp.m_general_data.cpu_core_name = iter->first;\n                }\n                theApp.m_cpu_temperature = iter->second;\n            }\n        }\n        else\n        {\n            theApp.m_cpu_temperature = -1;\n        }\n        //获取硬盘温度\n        if (!theApp.m_pMonitor->AllHDDTemperature().empty())\n        {\n            auto iter = theApp.m_pMonitor->AllHDDTemperature().find(theApp.m_general_data.hard_disk_name);\n            if (iter == theApp.m_pMonitor->AllHDDTemperature().end())\n            {\n                iter = theApp.m_pMonitor->AllHDDTemperature().begin();\n                theApp.m_general_data.hard_disk_name = iter->first;\n            }\n            theApp.m_hdd_temperature = iter->second;\n        }\n        else\n        {\n            theApp.m_hdd_temperature = -1;\n        }\n        //获取硬盘利用率\n        if (!theApp.m_pMonitor->AllHDDUsage().empty())\n        {\n            auto iter = theApp.m_pMonitor->AllHDDUsage().find(theApp.m_general_data.hard_disk_name);\n            if (iter == theApp.m_pMonitor->AllHDDUsage().end())\n            {\n                iter = theApp.m_pMonitor->AllHDDUsage().begin();\n                theApp.m_general_data.hard_disk_name = iter->first;\n            }\n            theApp.m_hdd_usage = iter->second;\n        }\n        else\n        {\n            theApp.m_hdd_usage = -1;\n        }\n    }\n    else\n    {\n        theApp.m_cpu_temperature = -1;\n        theApp.m_gpu_temperature = -1;\n        theApp.m_hdd_temperature = -1;\n        theApp.m_main_board_temperature = -1;\n        theApp.m_gpu_usage = -1;\n        theApp.m_hdd_usage = -1;\n    }\n#endif\n\n    //通知插件获取数据，以及向插件传递监控数据\n    for (const auto& plugin_info : theApp.m_plugins.GetPlugins())\n    {\n        if (plugin_info.plugin != nullptr)\n        {\n            plugin_info.plugin->DataRequired();\n            ITMPlugin::MonitorInfo monitor_info;\n            monitor_info.up_speed = theApp.m_out_speed;\n            monitor_info.down_speed = theApp.m_in_speed;\n            monitor_info.cpu_usage = theApp.m_cpu_usage;\n            monitor_info.memory_usage = theApp.m_memory_usage;\n            monitor_info.gpu_usage = theApp.m_gpu_usage;\n            monitor_info.hdd_usage = theApp.m_hdd_usage;\n            monitor_info.cpu_temperature = theApp.m_cpu_temperature;\n            monitor_info.gpu_temperature = theApp.m_gpu_temperature;\n            monitor_info.hdd_temperature = theApp.m_hdd_temperature;\n            monitor_info.cpu_freq = theApp.m_cpu_freq;\n            monitor_info.main_board_temperature = theApp.m_main_board_temperature;\n            plugin_info.plugin->OnMonitorInfo(monitor_info);\n        }\n    }\n\n    //}\n    pThis->m_monitor_time_cnt++;\n\n    //发送监控信息更新消息\n    pThis->SendMessage(WM_MONITOR_INFO_UPDATED);\n\n    return 0;\n}\n\nvoid CTrafficMonitorDlg::OnTimer(UINT_PTR nIDEvent)\n{\n    // TODO: 在此添加消息处理程序代码和/或调用默认值\n    if (nIDEvent == MONITOR_TIMER)\n    {\n        if (!m_is_monitor_thread_runing)    //确保线程已退出\n            AfxBeginThread(MonitorThreadCallback, (LPVOID)this);\n    }\n\n    if (nIDEvent == MAIN_TIMER)\n    {\n        if (m_first_start)      //这个if语句在程序启动后1秒执行\n        {\n            //将设置窗口置顶的处理放在这里是用于解决\n            //放在初始化函数中可能会出现设置置顶无效的问题\n            SetAlwaysOnTop();       //设置窗口置顶\n            SetMousePenetrate();    //设置鼠标穿透\n            if (theApp.m_cfg_data.m_hide_main_window)   //设置隐藏主窗口\n                ShowWindow(SW_HIDE);\n\n            //打开任务栏窗口\n            if (theApp.m_cfg_data.m_show_task_bar_wnd && m_tBarDlg == nullptr)\n                OpenTaskBarWnd();\n\n            //如果窗口的位置为(0, 0)，则在初始化时MoveWindow函数无效，此时再移动一次窗口\n            if (theApp.m_cfg_data.m_position_x == 0 && theApp.m_cfg_data.m_position_y == 0)\n            {\n                SetWindowPos(nullptr, theApp.m_cfg_data.m_position_x, theApp.m_cfg_data.m_position_y, 0, 0, SWP_NOZORDER | SWP_NOSIZE);\n            }\n            SetTransparency();              //重新设置窗口不透明度\n\n            m_first_start = false;\n        }\n\n        if (theApp.m_main_wnd_data.m_always_on_top && !theApp.m_cfg_data.m_hide_main_window)\n        {\n            //每隔1秒钟就判断一下前台窗口是否全屏\n            m_is_foreground_fullscreen = CCommon::IsForegroundFullscreen();\n            if (theApp.m_main_wnd_data.hide_main_wnd_when_fullscreen)       //当设置了全屏时隐藏悬浮窗时\n            {\n                if (m_is_foreground_fullscreen || theApp.m_cfg_data.m_hide_main_window)\n                    ShowWindow(SW_HIDE);\n                else\n                    ShowWindow(SW_RESTORE);\n            }\n            else            //如果没有设置全屏时隐藏悬浮窗，则如果有程序进入全屏状态，则设置一次窗口置顶\n            {\n                static bool last_foreground_fullscreen;\n                if (!last_foreground_fullscreen && m_is_foreground_fullscreen)\n                {\n                    SetAlwaysOnTop();\n                }\n                last_foreground_fullscreen = m_is_foreground_fullscreen;\n            }\n        }\n\n        if (!m_menu_popuped)\n        {\n            //程序启动后若干秒的时候根据设置重新执行“总是置顶”、“鼠标穿透”和“隐藏主窗口”的操作，防止设置没有生效\n            if (m_timer_cnt == 5 || m_timer_cnt == 9)\n            {\n                if (!theApp.m_cfg_data.m_hide_main_window)\n                {\n                    SetAlwaysOnTop();\n                    SetMousePenetrate();\n                }\n                else\n                {\n                    ShowWindow(SW_HIDE);\n                }\n            }\n\n            if (m_timer_cnt % 300 == 299 && !theApp.m_cfg_data.m_hide_main_window && theApp.m_main_wnd_data.m_always_on_top)\n            {\n                SetAlwaysOnTop();       //每5分钟执行一次设置窗口置顶\n            }\n        }\n\n        if (m_timer_cnt % 30 == 26)     //每隔30秒钟检测一次窗口位置，当窗口位置发生变化时保存设置\n        {\n            static int last_pos_x{ -1 }, last_pos_y{ -1 };\n            if (last_pos_x != theApp.m_cfg_data.m_position_x || last_pos_y != theApp.m_cfg_data.m_position_y)\n            {\n                theApp.SaveConfig();\n                last_pos_x = theApp.m_cfg_data.m_position_x;\n                last_pos_y = theApp.m_cfg_data.m_position_y;\n            }\n        }\n\n        if (m_timer_cnt % 2 == 1)       //每隔2秒钟获取一次屏幕区域\n        {\n            GetScreenSize();\n            CheckWindowPos();\n        }\n\n        //只有主窗口和任务栏窗口至少有一个显示时才执行下面的处理\n        if (!theApp.m_cfg_data.m_hide_main_window || theApp.m_cfg_data.m_show_task_bar_wnd)\n        {\n            //每隔10秒钟检测一次是否可以嵌入任务栏\n            if (IsTaskbarWndValid() && m_timer_cnt % 10 == 1)\n            {\n                if (m_tBarDlg->GetCannotInsertToTaskBar() && m_insert_to_taskbar_cnt < MAX_INSERT_TO_TASKBAR_CNT)\n                {\n                    CloseTaskBarWnd();\n                    OpenTaskBarWnd();\n                    m_insert_to_taskbar_cnt++;\n                    if (m_insert_to_taskbar_cnt == MAX_INSERT_TO_TASKBAR_CNT)\n                    {\n                        if (m_tBarDlg->GetCannotInsertToTaskBar() && m_cannot_intsert_to_task_bar_warning)      //确保提示信息只弹出一次\n                        {\n                            //写入错误日志\n                            CString info;\n                            info.LoadString(IDS_CONNOT_INSERT_TO_TASKBAR_ERROR_LOG);\n                            info.Replace(_T(\"<%cnt%>\"), CCommon::IntToString(m_insert_to_taskbar_cnt));\n                            info.Replace(_T(\"<%error_code%>\"), CCommon::IntToString(m_tBarDlg->GetErrorCode()));\n                            CCommon::WriteLog(info, theApp.m_log_path.c_str());\n                            //弹出错误信息\n                            MessageBox(CCommon::LoadText(IDS_CONNOT_INSERT_TO_TASKBAR, CCommon::IntToString(m_tBarDlg->GetErrorCode())), NULL, MB_ICONWARNING);\n                            m_cannot_intsert_to_task_bar_warning = false;\n                        }\n                    }\n                }\n                if (!m_tBarDlg->GetCannotInsertToTaskBar())\n                {\n                    m_insert_to_taskbar_cnt = 0;\n                }\n            }\n        }\n\n        //检查是否需要弹出鼠标提示\n        //setting_data: 消息提示的设置数据\n        //value: 当前的值\n        //last_value: 传递一个static或可以在此lambda表达式调用结束后继续存在的变量，用于保存上一次的值\n        //notify_time: 传递一个static或可以在此lambda表达式调用结束后继续存在的变量，用于记录上次弹出提示的时间（定时器触发次数）\n        //tip_str: 要提示的消息\n        auto checkNotifyTip = [&](GeneralSettingData::NotifyTipSettings setting_data, int value, int& last_value, int& notify_time, LPCTSTR tip_str)\n        {\n            if (setting_data.enable)\n            {\n                if (last_value < setting_data.tip_value && value >= setting_data.tip_value && (m_timer_cnt - notify_time > static_cast<unsigned int>(theApp.m_notify_interval)))\n                {\n                    ShowNotifyTip(CCommon::LoadText(_T(\"TrafficMonitor \"), IDS_NOTIFY), tip_str);\n                    notify_time = m_timer_cnt;\n                }\n                last_value = value;\n            }\n        };\n\n        //检查是否要弹出内存使用率超出提示\n        CString info;\n        info.Format(CCommon::LoadText(IDS_MEMORY_UDAGE_EXCEED, _T(\" %d%%!\")), theApp.m_memory_usage);\n        static int last_memory_usage;\n        static int memory_usage_notify_time{ -theApp.m_notify_interval };       //记录上次弹出提示时的时间\n        checkNotifyTip(theApp.m_general_data.memory_usage_tip, theApp.m_memory_usage, last_memory_usage, memory_usage_notify_time, info.GetString());\n\n        //检查是否要弹出CPU温度使用率超出提示\n        info.Format(CCommon::LoadText(IDS_CPU_TEMPERATURE_EXCEED, _T(\" %d°C!\")), static_cast<int>(theApp.m_cpu_temperature));\n        static int last_cpu_temp;\n        static int cpu_temp_notify_time{ -theApp.m_notify_interval };       //记录上次弹出提示时的时间\n        checkNotifyTip(theApp.m_general_data.cpu_temp_tip, theApp.m_cpu_temperature, last_cpu_temp, cpu_temp_notify_time, info.GetString());\n\n        //检查是否要弹出显卡温度使用率超出提示\n        info.Format(CCommon::LoadText(IDS_GPU_TEMPERATURE_EXCEED, _T(\" %d°C!\")), static_cast<int>(theApp.m_gpu_temperature));\n        static int last_gpu_temp;\n        static int gpu_temp_notify_time{ -theApp.m_notify_interval };       //记录上次弹出提示时的时间\n        checkNotifyTip(theApp.m_general_data.gpu_temp_tip, theApp.m_gpu_temperature, last_gpu_temp, gpu_temp_notify_time, info.GetString());\n\n        //检查是否要弹出硬盘温度使用率超出提示\n        info.Format(CCommon::LoadText(IDS_HDD_TEMPERATURE_EXCEED, _T(\" %d°C!\")), static_cast<int>(theApp.m_hdd_temperature));\n        static int last_hdd_temp;\n        static int hdd_temp_notify_time{ -theApp.m_notify_interval };       //记录上次弹出提示时的时间\n        checkNotifyTip(theApp.m_general_data.hdd_temp_tip, theApp.m_hdd_temperature, last_hdd_temp, hdd_temp_notify_time, info.GetString());\n\n        //检查是否要弹出主板温度使用率超出提示\n        info.Format(CCommon::LoadText(IDS_MBD_TEMPERATURE_EXCEED, _T(\" %d°C!\")), static_cast<int>(theApp.m_main_board_temperature));\n        static int last_main_board_temp;\n        static int main_board_temp_notify_time{ -theApp.m_notify_interval };        //记录上次弹出提示时的时间\n        checkNotifyTip(theApp.m_general_data.mainboard_temp_tip, theApp.m_main_board_temperature, last_main_board_temp, main_board_temp_notify_time, info.GetString());\n\n\n        //检查是否要弹出流量使用超出提示\n        if (theApp.m_general_data.traffic_tip_enable)\n        {\n            static __int64 last_today_traffic;\n            __int64 traffic_tip_value;\n            if (theApp.m_general_data.traffic_tip_unit == 0)\n                traffic_tip_value = static_cast<__int64>(theApp.m_general_data.traffic_tip_value) * 1024 * 1024;\n            else\n                traffic_tip_value = static_cast<__int64>(theApp.m_general_data.traffic_tip_value) * 1024 * 1024 * 1024;\n\n            __int64 today_traffic = theApp.m_today_up_traffic + theApp.m_today_down_traffic;\n            if (last_today_traffic < traffic_tip_value && today_traffic >= traffic_tip_value)\n            {\n                CString info = CCommon::LoadText(IDS_TODAY_TRAFFIC_EXCEED, CCommon::DataSizeToString(today_traffic));\n                ShowNotifyTip(CCommon::LoadText(_T(\"TrafficMonitor \"), IDS_NOTIFY), info.GetString());\n            }\n            last_today_traffic = today_traffic;\n        }\n\n        CWindowsSettingHelper::CheckWindows10LightTheme();        //每隔1秒钟检查一下当前系统是否为白色主题\n\n        //根据当前Win10颜色模式自动切换任务栏颜色\n        bool light_mode = CWindowsSettingHelper::IsWindows10LightTheme();\n        if (theApp.m_last_light_mode != light_mode)\n        {\n            theApp.m_last_light_mode = light_mode;\n            bool restart_taskbar_dlg{ false };\n            if (theApp.m_taskbar_data.auto_adapt_light_theme)\n            {\n                int style_index = CWindowsSettingHelper::IsWindows10LightTheme() ? theApp.m_taskbar_data.light_default_style : theApp.m_taskbar_data.dark_default_style;\n                theApp.m_taskbar_default_style.ApplyDefaultStyle(style_index, theApp.m_taskbar_data);\n                theApp.SaveConfig();\n                restart_taskbar_dlg = true;\n            }\n            bool is_taskbar_transparent{ CTaskbarDefaultStyle::IsTaskbarTransparent(theApp.m_taskbar_data) };\n            if (!is_taskbar_transparent)\n            {\n                CTaskbarDefaultStyle::SetTaskabrTransparent(false, theApp.m_taskbar_data);\n                restart_taskbar_dlg = true;\n            }\n            if (restart_taskbar_dlg && IsTaskbarWndValid())\n            {\n                //m_tBarDlg->ApplyWindowTransparentColor();\n                CloseTaskBarWnd();\n                OpenTaskBarWnd();\n\n                //写入调试日志\n                if (theApp.m_debug_log)\n                {\n                    CString log_str;\n                    log_str += _T(\"检测到 Windows10 深浅色变化。\\n\");\n                    log_str += _T(\"IsWindows10LightTheme: \");\n                    log_str += std::to_wstring(light_mode).c_str();\n                    log_str += _T(\"\\n\");\n                    log_str += _T(\"auto_adapt_light_theme: \");\n                    log_str += std::to_wstring(theApp.m_taskbar_data.auto_adapt_light_theme).c_str();\n                    log_str += _T(\"\\n\");\n                    log_str += _T(\"is_taskbar_transparent: \");\n                    log_str += std::to_wstring(is_taskbar_transparent).c_str();\n                    log_str += _T(\"\\n\");\n                    log_str += _T(\"taskbar_back_color: \");\n                    log_str += std::to_wstring(theApp.m_taskbar_data.back_color).c_str();\n                    log_str += _T(\"\\n\");\n                    log_str += _T(\"taskbar_transparent_color: \");\n                    log_str += std::to_wstring(theApp.m_taskbar_data.transparent_color).c_str();\n                    log_str += _T(\"\\n\");\n                    log_str += _T(\"taskbar_text_colors: \");\n                    for (const auto& item : theApp.m_taskbar_data.text_colors)\n                    {\n                        log_str += std::to_wstring(item.second.label).c_str();\n                        log_str += _T('|');\n                        log_str += std::to_wstring(item.second.value).c_str();\n                        log_str += _T(\", \");\n                    }\n                    log_str += _T(\"\\n\");\n                    CCommon::WriteLog(log_str, (theApp.m_config_dir + L\".\\\\debug.log\").c_str());\n                }\n            }\n\n            //根据当前Win10颜色模式自动切换通知区图标\n            if (theApp.m_cfg_data.m_notify_icon_auto_adapt)\n            {\n                int notify_icon_selected = theApp.m_cfg_data.m_notify_icon_selected;\n                theApp.AutoSelectNotifyIcon();\n                if (notify_icon_selected != theApp.m_cfg_data.m_notify_icon_selected)\n                {\n                    m_ntIcon.hIcon = theApp.m_notify_icons[theApp.m_cfg_data.m_notify_icon_selected];\n                    if (theApp.m_general_data.show_notify_icon)\n                    {\n                        DeleteNotifyIcon();\n                        AddNotifyIcon();\n                    }\n                }\n            }\n\n        }\n\n        //根据任务栏颜色自动设置任务栏窗口背景色\n        if (theApp.m_taskbar_data.auto_set_background_color && theApp.m_win_version.IsWindows8OrLater()\n            && IsTaskbarWndValid() && theApp.m_taskbar_data.transparent_color != 0\n            && !m_is_foreground_fullscreen)\n        {\n            CRect rect;\n            ::GetWindowRect(m_tBarDlg->GetSafeHwnd(), rect);\n            int pointx{ rect.left - 1 };\n            if (theApp.m_taskbar_data.tbar_wnd_on_left && m_tBarDlg->IsTasksbarOnTopOrBottom())\n                pointx = rect.right + 1;\n            int pointy = rect.bottom;\n            if (pointx < 0) pointx = 0;\n            if (pointx >= m_screen_size.cx) pointx = m_screen_size.cx - 1;\n            if (pointy < 0) pointy = 0;\n            if (pointy >= m_screen_size.cy) pointy = m_screen_size.cy - 1;\n            COLORREF color = ::GetPixel(m_desktop_dc, pointx, pointy);        //取任务栏窗口左侧1像素处的颜色作为背景色\n            if (!CCommon::IsColorSimilar(color, theApp.m_taskbar_data.back_color) && (/*CWindowsSettingHelper::IsWindows10LightTheme() ||*/ color != 0))\n            {\n                bool is_taskbar_transparent{ CTaskbarDefaultStyle::IsTaskbarTransparent(theApp.m_taskbar_data) };\n                theApp.m_taskbar_data.back_color = color;\n                CTaskbarDefaultStyle::SetTaskabrTransparent(is_taskbar_transparent, theApp.m_taskbar_data);\n                if (is_taskbar_transparent)\n                    m_tBarDlg->ApplyWindowTransparentColor();\n            }\n        }\n\n        //当检测到背景色和文字颜色都为黑色写入错误日志\n        static bool erro_log_write{ false };\n        if (theApp.m_taskbar_data.back_color == 0 && !theApp.m_taskbar_data.text_colors.empty() && theApp.m_taskbar_data.text_colors.begin()->second.label == 0)\n        {\n            if (!erro_log_write)\n            {\n                CString log_str;\n                log_str.Format(_T(\"检查到背景色和文字颜色都为黑色。IsWindows10LightTheme: %d, 系统启动时间：%d/%.2d/%.2d %.2d:%.2d:%.2d\"),\n                    light_mode, m_start_time.wYear, m_start_time.wMonth, m_start_time.wDay, m_start_time.wHour, m_start_time.wMinute, m_start_time.wSecond);\n                CCommon::WriteLog(log_str, theApp.m_log_path.c_str());\n                erro_log_write = true;\n            }\n        }\n        else\n        {\n            erro_log_write = false;\n        }\n\n        UpdateNotifyIconTip();\n\n        m_timer_cnt++;\n    }\n\n    if (nIDEvent == DELAY_TIMER)\n    {\n        AutoSelect();\n        KillTimer(DELAY_TIMER);\n    }\n\n    if (nIDEvent == TASKBAR_TIMER)\n    {\n        if (IsTaskbarWndValid())\n        {\n            m_tBarDlg->AdjustWindowPos();\n            m_tBarDlg->Invalidate(FALSE);\n        }\n    }\n\n    CDialog::OnTimer(nIDEvent);\n}\n\n\nvoid CTrafficMonitorDlg::OnRButtonUp(UINT nFlags, CPoint point)\n{\n    // TODO: 在此添加消息处理程序代码和/或调用默认值\n    CheckClickedItem(point);\n    if (m_clicked_item.is_plugin && m_clicked_item.plugin_item != nullptr)\n    {\n        ITMPlugin* plugin = theApp.m_plugins.GetPluginByItem(m_clicked_item.plugin_item);\n        if (plugin != nullptr && plugin->GetAPIVersion() >= 3)\n        {\n            if (m_clicked_item.plugin_item->OnMouseEvent(IPluginItem::MT_RCLICKED, point.x, point.y, (void*)GetSafeHwnd(), 0) != 0)\n                return;\n        }\n    }\n    //设置点击鼠标右键弹出菜单\n    CMenu* pContextMenu = theApp.m_main_menu.GetSubMenu(0); //获取第一个弹出菜单，所以第一个菜单必须有子菜单\n    CPoint point1;  //定义一个用于确定光标位置的位置\n    GetCursorPos(&point1);  //获取当前光标的位置，以便使得菜单可以跟随光标\n    //设置默认菜单项\n    switch (theApp.m_main_wnd_data.double_click_action)\n    {\n    case DoubleClickAction::CONNECTION_INFO:\n        pContextMenu->SetDefaultItem(ID_NETWORK_INFO);\n        break;\n        //case DoubleClickAction::HISTORY_TRAFFIC:\n        //  pContextMenu->SetDefaultItem(ID_TRAFFIC_HISTORY);\n        //  break;\n    case DoubleClickAction::SHOW_MORE_INFO:\n        pContextMenu->SetDefaultItem(ID_SHOW_CPU_MEMORY);\n        break;\n    case DoubleClickAction::OPTIONS:\n        pContextMenu->SetDefaultItem(ID_OPTIONS);\n        break;\n        //case DoubleClickAction::CHANGE_SKIN:\n        //  pContextMenu->SetDefaultItem(ID_CHANGE_SKIN);\n        //  break;\n    default:\n        pContextMenu->SetDefaultItem(-1);\n        break;\n    }\n    pContextMenu->TrackPopupMenu(TPM_LEFTALIGN | TPM_RIGHTBUTTON, point1.x, point1.y, this); //在指定位置显示弹出菜单\n\n    CDialog::OnRButtonUp(nFlags, point1);\n}\n\n\nvoid CTrafficMonitorDlg::OnLButtonDown(UINT nFlags, CPoint point)\n{\n    // TODO: 在此添加消息处理程序代码和/或调用默认值\n    //在未锁定窗口位置时允许通过点击窗口内部来拖动窗口\n    if (!theApp.m_main_wnd_data.m_lock_window_pos)\n        PostMessage(WM_NCLBUTTONDOWN, HTCAPTION, MAKELPARAM(point.x, point.y));\n    CDialog::OnLButtonDown(nFlags, point);\n}\n\n\nvoid CTrafficMonitorDlg::OnNetworkInfo()\n{\n    // TODO: 在此添加命令处理程序代码\n    //弹出“连接详情”对话框\n    CNetworkInfoDlg aDlg(m_connections, m_pIfTable->table, m_connection_selected);\n    ////向CNetworkInfoDlg类传递自启动以来已发送和接收的字节数\n    //aDlg.m_in_bytes = m_pIfTable->table[m_connections[m_connection_selected].index].dwInOctets - m_connections[m_connection_selected].in_bytes;\n    //aDlg.m_out_bytes = m_pIfTable->table[m_connections[m_connection_selected].index].dwOutOctets - m_connections[m_connection_selected].out_bytes;\n    aDlg.m_start_time = m_start_time;\n    aDlg.DoModal();\n    //SetAlwaysOnTop(); //由于在“连接详情”对话框内设置了取消窗口置顶，所有在对话框关闭后，重新设置窗口置顶\n    if (m_tBarDlg != nullptr)\n        m_tBarDlg->m_tool_tips.SetWindowPos(&wndTopMost, 0, 0, 0, 0, SWP_NOSIZE | SWP_NOMOVE);  //重新设置任务栏窗口的提示信息置顶\n}\n\n\nvoid CTrafficMonitorDlg::OnAlwaysOnTop()\n{\n    // TODO: 在此添加命令处理程序代码\n    theApp.m_main_wnd_data.m_always_on_top = !theApp.m_main_wnd_data.m_always_on_top;\n    SetAlwaysOnTop();\n    theApp.SaveConfig();\n}\n\n\nvoid CTrafficMonitorDlg::OnTransparency100()\n{\n    // TODO: 在此添加命令处理程序代码\n    theApp.m_cfg_data.m_transparency = 100;\n    SetTransparency();\n    theApp.SaveConfig();\n}\n\n\nvoid CTrafficMonitorDlg::OnTransparency80()\n{\n    // TODO: 在此添加命令处理程序代码\n    theApp.m_cfg_data.m_transparency = 80;\n    SetTransparency();\n    theApp.SaveConfig();\n}\n\n\nvoid CTrafficMonitorDlg::OnTransparency60()\n{\n    // TODO: 在此添加命令处理程序代码\n    theApp.m_cfg_data.m_transparency = 60;\n    SetTransparency();\n    theApp.SaveConfig();\n}\n\n\nvoid CTrafficMonitorDlg::OnTransparency40()\n{\n    // TODO: 在此添加命令处理程序代码\n    theApp.m_cfg_data.m_transparency = 40;\n    SetTransparency();\n    theApp.SaveConfig();\n}\n\n\nvoid CTrafficMonitorDlg::OnClose()\n{\n    // TODO: 在此添加消息处理程序代码和/或调用默认值\n    theApp.m_cannot_save_config_warning = true;\n    theApp.m_cannot_save_global_config_warning = true;\n    theApp.SaveConfig();    //退出前保存设置到ini文件\n    theApp.SaveGlobalConfig();\n    SaveHistoryTraffic();\n    BackupHistoryTrafficFile();\n\n    if (IsTaskbarWndValid())\n        m_tBarDlg->OnCancel();\n\n    //确保在退出前关闭所有窗口\n    for (const auto& item : CBaseDialog::AllUniqueHandels())\n    {\n        ::SendMessage(item.second, WM_COMMAND, IDCANCEL, 0);\n    }\n\n    CDialog::OnClose();\n}\n\n\nBOOL CTrafficMonitorDlg::OnCommand(WPARAM wParam, LPARAM lParam)\n{\n    // TODO: 在此添加专用代码和/或调用基类\n    UINT uMsg = LOWORD(wParam);\n    if (uMsg == ID_SELECT_ALL_CONNECTION)\n    {\n        theApp.m_cfg_data.m_select_all = true;\n        theApp.m_cfg_data.m_auto_select = false;\n        m_connection_change_flag = true;\n    }\n    //选择了“选择网络连接”子菜单中项目时的处理\n    if (uMsg == ID_SELETE_CONNECTION)   //选择了“自动选择”菜单项\n    {\n        AutoSelect();\n        theApp.m_cfg_data.m_auto_select = true;\n        theApp.m_cfg_data.m_select_all = false;\n        theApp.SaveConfig();\n        m_connection_change_flag = true;\n    }\n    if (uMsg > ID_SELECT_ALL_CONNECTION && uMsg <= ID_SELECT_ALL_CONNECTION + m_connections.size()) //选择了一个网络连接\n    {\n        m_connection_selected = uMsg - ID_SELECT_ALL_CONNECTION - 1;\n        theApp.m_cfg_data.m_connection_name = GetConnection(m_connection_selected).description_2;\n        m_connection_name_preferd = theApp.m_cfg_data.m_connection_name;\n        theApp.m_cfg_data.m_auto_select = false;\n        theApp.m_cfg_data.m_select_all = false;\n        theApp.SaveConfig();\n        m_connection_change_flag = true;\n    }\n#ifdef DEBUG\n    if (uMsg == ID_CMD_TEST)\n    {\n        CTest::TestCommand();\n    }\n#endif // DEBUG\n\n    return CDialog::OnCommand(wParam, lParam);\n    }\n\n\nvoid CTrafficMonitorDlg::OnInitMenu(CMenu* pMenu)\n{\n    CDialog::OnInitMenu(pMenu);\n\n    // TODO: 在此处添加消息处理程序代码\n    m_menu_popuped = true;\n\n    pMenu->CheckMenuItem(ID_ALWAYS_ON_TOP, MF_BYCOMMAND | (theApp.m_main_wnd_data.m_always_on_top ? MF_CHECKED : MF_UNCHECKED));\n    pMenu->CheckMenuItem(ID_LOCK_WINDOW_POS, MF_BYCOMMAND | (theApp.m_main_wnd_data.m_lock_window_pos ? MF_CHECKED : MF_UNCHECKED));\n    pMenu->CheckMenuItem(ID_SHOW_CPU_MEMORY, MF_BYCOMMAND | (theApp.m_cfg_data.m_show_more_info ? MF_CHECKED : MF_UNCHECKED));\n    pMenu->CheckMenuItem(ID_MOUSE_PENETRATE, MF_BYCOMMAND | (theApp.m_main_wnd_data.m_mouse_penetrate ? MF_CHECKED : MF_UNCHECKED));\n    pMenu->CheckMenuItem(ID_SHOW_TASK_BAR_WND, MF_BYCOMMAND | (theApp.m_cfg_data.m_show_task_bar_wnd ? MF_CHECKED : MF_UNCHECKED));\n    pMenu->CheckMenuItem(ID_SHOW_MAIN_WND, MF_BYCOMMAND | (!theApp.m_cfg_data.m_hide_main_window ? MF_CHECKED : MF_UNCHECKED));\n    pMenu->CheckMenuItem(ID_ALOW_OUT_OF_BORDER, MF_BYCOMMAND | (theApp.m_main_wnd_data.m_alow_out_of_border ? MF_CHECKED : MF_UNCHECKED));\n\n    //设置“选择连接”子菜单项中各单选项的选择状态\n    CMenu* select_connection_menu = theApp.m_main_menu.GetSubMenu(0)->GetSubMenu(0);\n    SetConnectionMenuState(select_connection_menu);\n\n    //设置“窗口不透明度”子菜单下各单选项的选择状态\n    switch (theApp.m_cfg_data.m_transparency)\n    {\n    case 100: pMenu->CheckMenuRadioItem(ID_TRANSPARENCY_100, ID_TRANSPARENCY_40, ID_TRANSPARENCY_100, MF_BYCOMMAND | MF_CHECKED); break;\n    case 80: pMenu->CheckMenuRadioItem(ID_TRANSPARENCY_100, ID_TRANSPARENCY_40, ID_TRANSPARENCY_80, MF_BYCOMMAND | MF_CHECKED); break;\n    case 60: pMenu->CheckMenuRadioItem(ID_TRANSPARENCY_100, ID_TRANSPARENCY_40, ID_TRANSPARENCY_60, MF_BYCOMMAND | MF_CHECKED); break;\n    case 40: pMenu->CheckMenuRadioItem(ID_TRANSPARENCY_100, ID_TRANSPARENCY_40, ID_TRANSPARENCY_40, MF_BYCOMMAND | MF_CHECKED); break;\n    default: break;\n    }\n\n    if (theApp.IsForceShowNotifyIcon())    //如果没有显示任务栏窗口，且隐藏了主窗口或设置了鼠标穿透，则禁用“显示通知区图标”菜单项\n        pMenu->EnableMenuItem(ID_SHOW_NOTIFY_ICON, MF_BYCOMMAND | MF_GRAYED);\n    else\n        pMenu->EnableMenuItem(ID_SHOW_NOTIFY_ICON, MF_BYCOMMAND | MF_ENABLED);\n\n    pMenu->EnableMenuItem(ID_SELECT_ALL_CONNECTION, MF_BYCOMMAND | (theApp.m_general_data.show_all_interface ? MF_GRAYED : MF_ENABLED));\n\n    pMenu->EnableMenuItem(ID_CHECK_UPDATE, MF_BYCOMMAND | (theApp.IsCheckingForUpdate() ? MF_GRAYED : MF_ENABLED));\n\n    //pMenu->SetDefaultItem(ID_NETWORK_INFO);\n}\n\n\nBOOL CTrafficMonitorDlg::PreTranslateMessage(MSG* pMsg)\n{\n    // TODO: 在此添加专用代码和/或调用基类\n    //屏蔽按回车键和ESC键退出\n    if (pMsg->message == WM_KEYDOWN && pMsg->wParam == VK_ESCAPE) return TRUE;\n    if (pMsg->message == WM_KEYDOWN && pMsg->wParam == VK_RETURN) return TRUE;\n\n    if (theApp.m_main_wnd_data.show_tool_tip && m_tool_tips.GetSafeHwnd())\n    {\n        m_tool_tips.RelayEvent(pMsg);\n    }\n\n    return CDialog::PreTranslateMessage(pMsg);\n}\n\n\nvoid CTrafficMonitorDlg::OnLockWindowPos()\n{\n    // TODO: 在此添加命令处理程序代码\n    theApp.m_main_wnd_data.m_lock_window_pos = !theApp.m_main_wnd_data.m_lock_window_pos;\n    theApp.SaveConfig();\n}\n\n\nvoid CTrafficMonitorDlg::OnMove(int x, int y)\n{\n    CDialog::OnMove(x, y);\n\n    // TODO: 在此处添加消息处理程序代码\n\n    if (!m_first_start)\n    {\n        theApp.m_cfg_data.m_position_x = x;\n        theApp.m_cfg_data.m_position_y = y;\n    }\n\n    ////确保窗口不会超出屏幕范围\n    //CheckWindowPos();\n}\n\n\n\nafx_msg LRESULT CTrafficMonitorDlg::OnNotifyIcon(WPARAM wParam, LPARAM lParam)\n{\n    bool dialog_exist{ false };\n    HWND handle{};\n    if (lParam == WM_LBUTTONDOWN || lParam == WM_RBUTTONUP || lParam == WM_LBUTTONDBLCLK)\n    {\n        for (const auto& item : CBaseDialog::AllUniqueHandels())\n        {\n            if (IsWindow(item.second))\n            {\n                dialog_exist = true;\n                handle = item.second;\n                break;\n            }\n        }\n    }\n\n    if (lParam == WM_LBUTTONDOWN)\n    {\n        if (!theApp.m_cfg_data.m_hide_main_window)\n        {\n            if (dialog_exist)       //有打开的对话框时，点击通知区图标后将焦点设置到对话框\n            {\n                ::SetForegroundWindow(handle);\n            }\n            else        //没有打开的对话框时，则显示主窗口\n            {\n                ShowWindow(SW_RESTORE);\n                theApp.m_cfg_data.m_hide_main_window = false;\n                SetForegroundWindow();\n                SetAlwaysOnTop();\n                CheckWindowPos();\n                theApp.SaveConfig();\n            }\n        }\n    }\n    if (lParam == WM_RBUTTONUP)\n    {\n        if (dialog_exist)       //有打开的对话框时，点击通知区图标后将焦点设置到对话框\n        {\n            ::SetForegroundWindow(handle);\n        }\n        else\n        {\n            //在通知区点击右键弹出右键菜单\n            if (IsTaskbarWndValid())        //如果显示了任务栏窗口，则在右击了通知区图标后将焦点设置到任务栏窗口\n                m_tBarDlg->SetForegroundWindow();\n            else                //否则将焦点设置到主窗口\n                SetForegroundWindow();\n            CPoint point1;  //定义一个用于确定光标位置的位置\n            GetCursorPos(&point1);  //获取当前光标的位置，以便使得菜单可以跟随光标\n            theApp.m_main_menu.GetSubMenu(0)->SetDefaultItem(-1);       //设置没有默认菜单项\n            theApp.m_main_menu.GetSubMenu(0)->TrackPopupMenu(TPM_LEFTALIGN | TPM_RIGHTBUTTON, point1.x, point1.y, this); //在指定位置显示弹出菜单\n\n            CheckWindowPos();\n        }\n    }\n    if (lParam == WM_LBUTTONDBLCLK)\n    {\n        if (dialog_exist)       //有打开的对话框时，点击通知区图标后将焦点设置到对话框\n        {\n            ::SetForegroundWindow(handle);\n        }\n        else        //没有打开的对话框时，则显示主窗口\n        {\n            ShowWindow(SW_RESTORE);\n            theApp.m_cfg_data.m_hide_main_window = false;\n            SetForegroundWindow();\n            SetAlwaysOnTop();\n            CheckWindowPos();\n            theApp.SaveConfig();\n        }\n    }\n    return 0;\n}\n\n\nvoid CTrafficMonitorDlg::OnShowNotifyIcon()\n{\n    // TODO: 在此添加命令处理程序代码\n    if (theApp.m_general_data.show_notify_icon)\n    {\n        DeleteNotifyIcon();\n        theApp.m_general_data.show_notify_icon = false;\n    }\n    else\n    {\n        AddNotifyIcon();\n        theApp.m_general_data.show_notify_icon = true;\n    }\n    theApp.SaveConfig();\n}\n\n\nvoid CTrafficMonitorDlg::OnDestroy()\n{\n    CDialog::OnDestroy();\n\n    //程序退出时删除通知栏图标\n    ::Shell_NotifyIcon(NIM_DELETE, &m_ntIcon);\n    // TODO: 在此处添加消息处理程序代码\n}\n\n\nvoid CTrafficMonitorDlg::OnShowCpuMemory()\n{\n    // TODO: 在此添加命令处理程序代码\n    CRect rect;\n    GetWindowRect(rect);\n    theApp.m_cfg_data.m_show_more_info = !theApp.m_cfg_data.m_show_more_info;\n    if (theApp.m_cfg_data.m_show_more_info)\n    {\n        rect.right = rect.left + m_skin.GetLayoutInfo().layout_l.width;\n        rect.bottom = rect.top + m_skin.GetLayoutInfo().layout_l.width;\n        MoveWindow(rect);\n        CheckWindowPos();\n    }\n    else\n    {\n        rect.right = rect.left + m_skin.GetLayoutInfo().layout_s.width;\n        rect.bottom = rect.top + m_skin.GetLayoutInfo().layout_s.height;\n        MoveWindow(rect);\n        CheckWindowPos();\n    }\n    LoadBackGroundImage();\n    SetItemPosition();\n    Invalidate(FALSE);\n    theApp.SaveConfig();\n}\n\n\n//任务栏窗口切换显示CPU和内存利用率时的处理\nvoid CTrafficMonitorDlg::OnShowCpuMemory2()\n{\n    // TODO: 在此添加命令处理程序代码\n    if (m_tBarDlg != nullptr)\n    {\n        bool show_cpu_memory = ((theApp.m_taskbar_data.m_tbar_display_item & TDI_CPU) || (theApp.m_taskbar_data.m_tbar_display_item & TDI_MEMORY));\n        if (show_cpu_memory)\n        {\n            theApp.m_taskbar_data.m_tbar_display_item &= ~TDI_CPU;\n            theApp.m_taskbar_data.m_tbar_display_item &= ~TDI_MEMORY;\n        }\n        else\n        {\n            theApp.m_taskbar_data.m_tbar_display_item |= TDI_CPU;\n            theApp.m_taskbar_data.m_tbar_display_item |= TDI_MEMORY;\n        }\n        //theApp.m_cfg_data.m_tbar_show_cpu_memory = !theApp.m_cfg_data.m_tbar_show_cpu_memory;\n        //切换显示CPU和内存利用率时，删除任务栏窗口，再重新显示\n        CloseTaskBarWnd();\n        OpenTaskBarWnd();\n    }\n}\n\n\nvoid CTrafficMonitorDlg::OnMousePenetrate()\n{\n    // TODO: 在此添加命令处理程序代码\n    theApp.m_main_wnd_data.m_mouse_penetrate = !theApp.m_main_wnd_data.m_mouse_penetrate;\n    SetMousePenetrate();\n    if (!theApp.m_general_data.show_notify_icon && theApp.IsForceShowNotifyIcon())   //鼠标穿透时，如果通知图标没有显示，则将它显示出来，否则无法呼出右键菜单\n    {\n        //添加通知栏图标\n        AddNotifyIcon();\n        theApp.m_general_data.show_notify_icon = true;\n    }\n\n    //设置鼠标穿透后，弹出消息提示用户如何关闭鼠标穿透\n    if (theApp.m_main_wnd_data.m_mouse_penetrate && theApp.m_show_mouse_panetrate_tip)\n    {\n        if (MessageBox(CCommon::LoadText(IDS_MOUSE_PENETRATE_TIP_INFO), NULL, MB_ICONINFORMATION | MB_OKCANCEL) == IDCANCEL)        //点击“取消”后不再提示\n        {\n            theApp.m_show_mouse_panetrate_tip = false;\n        }\n    }\n\n    theApp.SaveConfig();\n}\n\n\nvoid CTrafficMonitorDlg::OnShowTaskBarWnd()\n{\n    // TODO: 在此添加命令处理程序代码\n    if (m_tBarDlg != nullptr)\n    {\n        CloseTaskBarWnd();\n    }\n    if (!theApp.m_cfg_data.m_show_task_bar_wnd)\n    {\n        theApp.m_cfg_data.m_show_task_bar_wnd = true;\n        OpenTaskBarWnd();\n    }\n    else\n    {\n        theApp.m_cfg_data.m_show_task_bar_wnd = false;\n        //关闭任务栏窗口后，如果没有显示通知区图标，且没有显示主窗口或设置了鼠标穿透，则将通知区图标显示出来\n        if (!theApp.m_general_data.show_notify_icon && theApp.IsForceShowNotifyIcon())\n        {\n            AddNotifyIcon();\n            theApp.m_general_data.show_notify_icon = true;\n        }\n    }\n    theApp.SaveConfig();\n}\n\n\nvoid CTrafficMonitorDlg::OnAppAbout()\n{\n    // TODO: 在此添加命令处理程序代码\n    //弹出“关于”对话框\n    CAboutDlg aDlg;\n    aDlg.DoModal();\n}\n\n\n//当资源管理器重启时会触发此消息\nLRESULT CTrafficMonitorDlg::OnTaskBarCreated(WPARAM wParam, LPARAM lParam)\n{\n    if (m_tBarDlg != nullptr)\n    {\n        CloseTaskBarWnd();\n        if (theApp.m_general_data.show_notify_icon)\n        {\n            //重新添加通知栏图标\n            ::Shell_NotifyIcon(NIM_ADD, &m_ntIcon);\n        }\n        OpenTaskBarWnd();\n    }\n    else\n    {\n        if (theApp.m_general_data.show_notify_icon)\n            ::Shell_NotifyIcon(NIM_ADD, &m_ntIcon);\n    }\n    return LRESULT();\n}\n\n\nvoid CTrafficMonitorDlg::OnShowMainWnd()\n{\n    // TODO: 在此添加命令处理程序代码\n    if (!theApp.m_cfg_data.m_hide_main_window)\n    {\n        ShowWindow(SW_HIDE);\n        theApp.m_cfg_data.m_hide_main_window = true;\n        //隐藏主窗口后，如果没有显示通知栏图标，则将其显示出来\n        if (!theApp.m_general_data.show_notify_icon && theApp.IsForceShowNotifyIcon())\n        {\n            AddNotifyIcon();\n            theApp.m_general_data.show_notify_icon = true;\n        }\n    }\n    else\n    {\n        ShowWindow(SW_RESTORE);\n        theApp.m_cfg_data.m_hide_main_window = false;\n    }\n    theApp.SaveConfig();\n}\n\n\nvoid CTrafficMonitorDlg::OnChangeSkin()\n{\n    // TODO: 在此添加命令处理程序代码\n    CSkinDlg skinDlg;\n    //初始化CSkinDlg对象的数据\n    skinDlg.m_skins = m_skins;\n    skinDlg.m_skin_selected = m_skin_selected;\n    skinDlg.m_pFont = &m_font;\n    if (skinDlg.DoModal() == IDOK)\n    {\n        m_skin_selected = skinDlg.m_skin_selected;\n        theApp.m_cfg_data.m_skin_name = m_skins[m_skin_selected];\n        //获取皮肤布局\n        LoadSkinLayout();\n        //载入背景图片\n        LoadBackGroundImage();\n        //获取皮肤的文字颜色\n        theApp.m_main_wnd_data.specify_each_item_color = skinDlg.GetSkinData().GetSkinInfo().specify_each_item_color;\n        int i{};\n        for (const auto& item : theApp.m_plugins.AllDisplayItemsWithPlugins())\n        {\n            theApp.m_main_wnd_data.text_colors[item] = skinDlg.GetSkinData().GetSkinInfo().TextColor(i);\n            i++;\n        }\n        //SetTextColor();\n        //获取皮肤的字体\n        if (theApp.m_general_data.allow_skin_cover_font)\n        {\n            if (!skinDlg.GetSkinData().GetSkinInfo().font_info.name.IsEmpty())\n            {\n                theApp.m_main_wnd_data.font.name = skinDlg.GetSkinData().GetSkinInfo().font_info.name;\n                theApp.m_main_wnd_data.font.bold = skinDlg.GetSkinData().GetSkinInfo().font_info.bold;\n                theApp.m_main_wnd_data.font.italic = skinDlg.GetSkinData().GetSkinInfo().font_info.italic;\n                theApp.m_main_wnd_data.font.underline = skinDlg.GetSkinData().GetSkinInfo().font_info.underline;\n                theApp.m_main_wnd_data.font.strike_out = skinDlg.GetSkinData().GetSkinInfo().font_info.strike_out;\n            }\n            if (skinDlg.GetSkinData().GetSkinInfo().font_info.size >= MIN_FONT_SIZE && skinDlg.GetSkinData().GetSkinInfo().font_info.size <= MAX_FONT_SIZE)\n                theApp.m_main_wnd_data.font.size = skinDlg.GetSkinData().GetSkinInfo().font_info.size;\n            SetTextFont();\n        }\n        //获取项目的显示文本\n        if (theApp.m_general_data.allow_skin_cover_text && !skinDlg.GetSkinData().GetLayoutInfo().no_label)\n        {\n            theApp.m_main_wnd_data.disp_str = skinDlg.GetSkinData().GetSkinInfo().display_text;\n        }\n        SetItemPosition();\n        Invalidate(FALSE);      //更换皮肤后立即刷新窗口信息\n        theApp.SaveConfig();\n    }\n}\n\n\n\nvoid CTrafficMonitorDlg::OnTrafficHistory()\n{\n    // TODO: 在此添加命令处理程序代码\n    CHistoryTrafficDlg historyDlg(m_history_traffic.GetTraffics());\n    historyDlg.DoModal();\n}\n\n\nvoid CTrafficMonitorDlg::OnMouseMove(UINT nFlags, CPoint point)\n{\n    // TODO: 在此添加消息处理程序代码和/或调用默认值\n\n    CDialog::OnMouseMove(nFlags, point);\n}\n\n\nvoid CTrafficMonitorDlg::OnLButtonDblClk(UINT nFlags, CPoint point)\n{\n    // TODO: 在此添加消息处理程序代码和/或调用默认值\n    CheckClickedItem(point);\n    if (m_clicked_item.is_plugin && m_clicked_item.plugin_item != nullptr)\n    {\n        ITMPlugin* plugin = theApp.m_plugins.GetPluginByItem(m_clicked_item.plugin_item);\n        if (plugin != nullptr && plugin->GetAPIVersion() >= 3)\n        {\n            if (m_clicked_item.plugin_item->OnMouseEvent(IPluginItem::MT_DBCLICKED, point.x, point.y, (void*)GetSafeHwnd(), 0) != 0)\n                return;\n        }\n    }\n\n    switch (theApp.m_main_wnd_data.double_click_action)\n    {\n    case DoubleClickAction::CONNECTION_INFO:\n        OnNetworkInfo();            //双击后弹出“连接详情”对话框\n        break;\n    case DoubleClickAction::HISTORY_TRAFFIC:\n        OnTrafficHistory();         //双击后弹出“历史流量统计”对话框\n        break;\n    case DoubleClickAction::SHOW_MORE_INFO:\n        OnShowCpuMemory();          //切换显示CPU和内存利用率\n        break;\n    case DoubleClickAction::OPTIONS:\n        OnOptions();                //双击后弹出“选项设置”对话框\n        break;\n    case DoubleClickAction::TASK_MANAGER:\n        ShellExecuteW(NULL, _T(\"open\"), (theApp.m_system_dir + L\"\\\\Taskmgr.exe\").c_str(), NULL, NULL, SW_NORMAL);       //打开任务管理器\n        break;\n    case DoubleClickAction::SEPCIFIC_APP:\n        ShellExecuteW(NULL, _T(\"open\"), (theApp.m_main_wnd_data.double_click_exe).c_str(), NULL, NULL, SW_NORMAL);  //打开指定程序，默认任务管理器\n        break;\n    case DoubleClickAction::CHANGE_SKIN:\n        OnChangeSkin();             //双击后弹出“更换皮肤”对话框\n        break;\n    default:\n        break;\n    }\n    CDialog::OnLButtonDblClk(nFlags, point);\n}\n\n\nvoid CTrafficMonitorDlg::OnOptions()\n{\n    // TODO: 在此添加命令处理程序代码\n    _OnOptions(0);\n}\n\n\n//通过任务栏窗口的右键菜单打开“选项”对话框\nvoid CTrafficMonitorDlg::OnOptions2()\n{\n    // TODO: 在此添加命令处理程序代码\n    //判断任务栏窗口中点击的项目是否是插件项目\n    if (IsTaskbarWndValid() && m_tBarDlg->GetClickedItem().is_plugin)\n    {\n        //找到对应的插件\n        ITMPlugin* plugin = theApp.m_plugins.GetPluginByItem(m_tBarDlg->GetClickedItem().plugin_item);\n        if (plugin != nullptr)\n        {\n            //显示插件的选项设置\n            auto rtn = plugin->ShowOptionsDialog(GetSafeHwnd());\n            if (rtn == ITMPlugin::OR_OPTION_CHANGED)    //选项设置有更改，重新打开任务栏窗口\n            {\n                CloseTaskBarWnd();\n                OpenTaskBarWnd();\n            }\n            if (rtn != ITMPlugin::OR_OPTION_NOT_PROVIDED)\n                return;\n        }\n    }\n\n    _OnOptions(1);\n}\n\n\nafx_msg LRESULT CTrafficMonitorDlg::OnExitmenuloop(WPARAM wParam, LPARAM lParam)\n{\n    m_menu_popuped = false;\n    return 0;\n}\n\n\nvoid CTrafficMonitorDlg::OnChangeNotifyIcon()\n{\n    // TODO: 在此添加命令处理程序代码\n    CIconSelectDlg dlg(theApp.m_cfg_data.m_notify_icon_selected);\n    dlg.SetAutoAdaptNotifyIcon(theApp.m_cfg_data.m_notify_icon_auto_adapt);\n    if (dlg.DoModal() == IDOK)\n    {\n        theApp.m_cfg_data.m_notify_icon_selected = dlg.GetIconSelected();\n        theApp.m_cfg_data.m_notify_icon_auto_adapt = dlg.AutoAdaptNotifyIcon();\n        m_ntIcon.hIcon = theApp.m_notify_icons[theApp.m_cfg_data.m_notify_icon_selected];\n        if (theApp.m_cfg_data.m_notify_icon_auto_adapt)\n            theApp.AutoSelectNotifyIcon();\n        if (theApp.m_general_data.show_notify_icon)\n        {\n            DeleteNotifyIcon();\n            AddNotifyIcon();\n        }\n        theApp.SaveConfig();\n    }\n}\n\n\nvoid CTrafficMonitorDlg::OnAlowOutOfBorder()\n{\n    // TODO: 在此添加命令处理程序代码\n    theApp.m_main_wnd_data.m_alow_out_of_border = !theApp.m_main_wnd_data.m_alow_out_of_border;\n    CheckWindowPos();\n}\n\n\nvoid CTrafficMonitorDlg::OnCheckUpdate()\n{\n    // TODO: 在此添加命令处理程序代码\n    theApp.CheckUpdateInThread(true);\n}\n\n\nafx_msg LRESULT CTrafficMonitorDlg::OnTaskbarMenuPopedUp(WPARAM wParam, LPARAM lParam)\n{\n    //设置“选择连接”子菜单项中各单选项的选择状态\n    CMenu* select_connection_menu = theApp.m_taskbar_menu.GetSubMenu(0)->GetSubMenu(0);\n    SetConnectionMenuState(select_connection_menu);\n    return 0;\n}\n\n\n//任务栏窗口切换显示网速时的处理\nvoid CTrafficMonitorDlg::OnShowNetSpeed()\n{\n    // TODO: 在此添加命令处理程序代码\n    if (m_tBarDlg != nullptr)\n    {\n        bool show_net_speed = ((theApp.m_taskbar_data.m_tbar_display_item & TDI_UP) || (theApp.m_taskbar_data.m_tbar_display_item & TDI_DOWN));\n        if (show_net_speed)\n        {\n            theApp.m_taskbar_data.m_tbar_display_item &= ~TDI_UP;\n            theApp.m_taskbar_data.m_tbar_display_item &= ~TDI_DOWN;\n        }\n        else\n        {\n            theApp.m_taskbar_data.m_tbar_display_item |= TDI_UP;\n            theApp.m_taskbar_data.m_tbar_display_item |= TDI_DOWN;\n        }\n        CloseTaskBarWnd();\n        OpenTaskBarWnd();\n    }\n}\n\n\nBOOL CTrafficMonitorDlg::OnQueryEndSession()\n{\n    if (!CDialog::OnQueryEndSession())\n        return FALSE;\n\n    // TODO:  在此添加专用的查询结束会话代码\n    theApp.SaveConfig();\n    theApp.SaveGlobalConfig();\n    SaveHistoryTraffic();\n    BackupHistoryTrafficFile();\n\n    if (theApp.m_debug_log)\n    {\n        CCommon::WriteLog(_T(\"TrafficMonitor进程已被终止，设置已保存。\"), (theApp.m_config_dir + L\".\\\\debug.log\").c_str());\n    }\n\n    return TRUE;\n}\n\n\nvoid CTrafficMonitorDlg::OnPaint()\n{\n    CPaintDC dc(this); // device context for painting\n                       // TODO: 在此处添加消息处理程序代码\n                       // 不为绘图消息调用 CDialog::OnPaint()\n    m_skin.DrawInfo(&dc, theApp.m_cfg_data.m_show_more_info, m_font);\n}\n\n\nafx_msg LRESULT CTrafficMonitorDlg::OnDpichanged(WPARAM wParam, LPARAM lParam)\n{\n    int dpi = LOWORD(wParam);\n    theApp.SetDPI(dpi);\n    if (IsTaskbarWndValid())\n    {\n        //根据新的DPI重新设置任务栏窗口字体\n        m_tBarDlg->SetTextFont();\n    }\n\n    LoadSkinLayout();   //根据当前选择的皮肤获取布局数据\n    SetItemPosition();  //初始化窗口位置\n    LoadBackGroundImage();\n    SetTextFont();      //重新加载字体\n    Invalidate(FALSE);  //重绘界面\n\n    return 0;\n}\n\n\nafx_msg LRESULT CTrafficMonitorDlg::OnTaskbarWndClosed(WPARAM wParam, LPARAM lParam)\n{\n    theApp.m_cfg_data.m_show_task_bar_wnd = false;\n    //关闭任务栏窗口后，如果没有显示通知区图标，且没有显示主窗口或设置了鼠标穿透，则将通知区图标显示出来\n    if (!theApp.m_general_data.show_notify_icon && theApp.IsForceShowNotifyIcon())\n    {\n        AddNotifyIcon();\n        theApp.m_general_data.show_notify_icon = true;\n    }\n    return 0;\n}\n\n\n\nafx_msg LRESULT CTrafficMonitorDlg::OnMonitorInfoUpdated(WPARAM wParam, LPARAM lParam)\n{\n    Invalidate(FALSE);      //刷新窗口信息\n\n    //更新鼠标提示\n    if (theApp.m_main_wnd_data.show_tool_tip && m_tool_tips.GetSafeHwnd() != NULL)\n    {\n        CString tip_info;\n        tip_info = GetMouseTipsInfo();\n        m_tool_tips.UpdateTipText(tip_info, this);\n    }\n    //更新任务栏窗口鼠标提示\n    if (IsTaskbarWndValid())\n        m_tBarDlg->UpdateToolTips();\n    return 0;\n}\n\n\nLRESULT CTrafficMonitorDlg::OnDisplaychange(WPARAM wParam, LPARAM lParam)\n{\n    GetScreenSize();\n    CheckWindowPos(true);\n    return 0;\n}\n\n\nvoid CTrafficMonitorDlg::OnExitSizeMove()\n{\n    // TODO: 在此添加消息处理程序代码和/或调用默认值\n    CheckWindowPos();\n\n    CDialog::OnExitSizeMove();\n}\n\n\nvoid CTrafficMonitorDlg::OnPluginManage()\n{\n    // TODO: 在此添加命令处理程序代码\n    CPluginManagerDlg dlg;\n    dlg.DoModal();\n}\n\nLRESULT CTrafficMonitorDlg::OnReopenTaksbarWnd(WPARAM wParam, LPARAM lParam)\n{\n    CloseTaskBarWnd();\n    OpenTaskBarWnd();\n    return 0;\n}\n\n\nvoid CTrafficMonitorDlg::OnOpenTaskManager()\n{\n    ShellExecuteW(NULL, _T(\"open\"), (theApp.m_system_dir + L\"\\\\Taskmgr.exe\").c_str(), NULL, NULL, SW_NORMAL);       //打开任务管理器\n}\n\n\nafx_msg LRESULT CTrafficMonitorDlg::OnSettingsApplied(WPARAM wParam, LPARAM lParam)\n{\n    COptionsDlg* pOptionsDlg = (COptionsDlg*)wParam;\n    if (pOptionsDlg != nullptr)\n    {\n        ApplySettings(*pOptionsDlg);\n    }\n    return 0;\n}\n\n\nvoid CTrafficMonitorDlg::OnDisplaySettings()\n{\n    // TODO: 在此添加命令处理程序代码\n    CSetItemOrderDlg dlg;\n    dlg.SetItemOrder(theApp.m_taskbar_data.item_order.GetItemOrderConst());\n    dlg.SetDisplayItem(theApp.m_taskbar_data.m_tbar_display_item);\n    dlg.SetPluginDisplayItem(theApp.m_taskbar_data.plugin_display_item);\n    if (dlg.DoModal() == IDOK)\n    {\n        theApp.m_taskbar_data.item_order.SetOrder(dlg.GetItemOrder());\n        theApp.m_taskbar_data.m_tbar_display_item = dlg.GetDisplayItem();\n        theApp.m_taskbar_data.plugin_display_item = dlg.GetPluginDisplayItem();\n        CloseTaskBarWnd();\n        OpenTaskBarWnd();\n    }\n}\n\n\nvoid CTrafficMonitorDlg::OnLButtonUp(UINT nFlags, CPoint point)\n{\n    // TODO: 在此添加消息处理程序代码和/或调用默认值\n    CheckClickedItem(point);\n    if (m_clicked_item.is_plugin && m_clicked_item.plugin_item != nullptr)\n    {\n        ITMPlugin* plugin = theApp.m_plugins.GetPluginByItem(m_clicked_item.plugin_item);\n        if (plugin != nullptr && plugin->GetAPIVersion() >= 3)\n        {\n            if (m_clicked_item.plugin_item->OnMouseEvent(IPluginItem::MT_LCLICKED, point.x, point.y, (void*)GetSafeHwnd(), 0) != 0)\n                return;\n        }\n    }\n\n    CDialog::OnLButtonUp(nFlags, point);\n}\n\n\nvoid CTrafficMonitorDlg::OnRefreshConnectionList()\n{\n    IniConnection();\n\n}\n"
  },
  {
    "path": "TrafficMonitor/TrafficMonitorDlg.h",
    "content": "﻿\n// TrafficMonitorDlg.h : 头文件\n//\n\n#pragma once\n#pragma comment (lib, \"iphlpapi.lib\")\n\n#include \"NetworkInfoDlg.h\"\n#include \"afxwin.h\"\n#include \"StaticEx.h\"\n#include \"Common.h\"\n#include \"TaskBarDlg.h\"\n#include \"SkinDlg.h\"\n#include \"HistoryTrafficDlg.h\"\n#include \"OptionsDlg.h\"\n#include \"PictureStatic.h\"\n#include \"IconSelectDlg.h\"\n#include \"DrawCommon.h\"\n#include \"IniHelper.h\"\n#include \"LinkStatic.h\"\n#include \"AdapterCommon.h\"\n#include \"AboutDlg.h\"\n#include \"CPUUsage.h\"\n#include \"HistoryTrafficFile.h\"\n#include \"HighResolutionTimer.h\"\n\n// CTrafficMonitorDlg 对话框\nclass CTrafficMonitorDlg : public CDialog\n{\n    // 构造\npublic:\n    CTrafficMonitorDlg(CWnd* pParent = NULL);   // 标准构造函数\n    ~CTrafficMonitorDlg();\n\n    // 对话框数据\n#ifdef AFX_DESIGN_TIME\n    enum { IDD = IDD_TRAFFICMONITOR_DIALOG };\n#endif\n\nprotected:\n    virtual void DoDataExchange(CDataExchange* pDX);    // DDX/DDV 支持\n\n\n// 实现\nprotected:\n    HICON m_hIcon;\n    NOTIFYICONDATA m_ntIcon;    //通知区域图标\n    CTaskBarDlg* m_tBarDlg{};     //任务栏窗口的指针\n\n    vector<NetWorkConection> m_connections; //保存获取到的要显示到“选择网卡”菜单项中的所有网络连接\n    MIB_IFTABLE* m_pIfTable;\n    DWORD m_dwSize{};\t//m_pIfTable的大小\n    int m_connection_selected{ 0 }; //要显示流量的连接的序号\n    unsigned __int64 m_in_bytes{};        //当前已接收的字节数\n    unsigned __int64 m_out_bytes{};   //当前已发送的字节数\n    unsigned __int64 m_last_in_bytes{}; //上次已接收的字节数\n    unsigned __int64 m_last_out_bytes{};    //上次已发送的字节数\n\n    CCPUUsage m_cpu_usage;\n\n    bool m_first_start{ true };     //初始时为true，在定时器第一次启动后置为flase\n\n    // https://www.jianshu.com/p/9d4b68cdbd99\n    struct Monitors\n    {\n        std::vector<MONITORINFO> monitorinfos;\n\n        static BOOL CALLBACK MonitorEnum(HMONITOR hMon, HDC hdc, LPRECT lprcMonitor, LPARAM pData)\n        {\n            MONITORINFO iMonitor;\n            iMonitor.cbSize = sizeof(MONITORINFO);\n            GetMonitorInfo(hMon, &iMonitor);\n\n            Monitors* pThis = reinterpret_cast<Monitors*>(pData);\n            pThis->monitorinfos.push_back(iMonitor);\n            return TRUE;\n        }\n\n        Monitors()\n        {\n            EnumDisplayMonitors(0, 0, MonitorEnum, (LPARAM)this);\n        }\n    };\n\n    //CRect m_screen_rect;        //屏幕的范围（不包含任务栏）\n    vector<CRect> m_screen_rects;       //所有屏幕的范围（不包含任务栏）\n    vector<CRect> m_last_screen_rects;       //上一次所有屏幕的范围（不包含任务栏）\n    CSize m_screen_size;        //屏幕的大小（包含任务栏）\n    CSkinFile m_skin;\n    CommonDisplayItem m_clicked_item;           //鼠标点击的项目\n\n    CFont m_font;           //字体\n\n    int m_restart_cnt{ -1 };    //重新初始化次数\n    unsigned int m_timer_cnt{};     //定时器触发次数（自程序启动以来的秒数）\n    unsigned int m_monitor_time_cnt{};\n    int m_zero_speed_cnt{}; //如果检测不到网速，该变量就会自加\n    int m_insert_to_taskbar_cnt{};  //用来统计尝试嵌入任务栏的次数\n    int m_cannot_intsert_to_task_bar_warning{ true };   //指示是否会在无法嵌入任务栏时弹出提示框\n\n    static unsigned int m_WM_TASKBARCREATED;    //任务栏重启消息\n\n    vector<wstring> m_skins;    //储存皮肤文件的路径\n    int m_skin_selected{};      //选择的皮肤序号\n\n    SYSTEMTIME m_start_time;    //程序启动时的时间\n    CHistoryTrafficFile m_history_traffic{ theApp.m_history_traffic_path }; //储存历史流量\n\n    CToolTipCtrl m_tool_tips;\n\n    bool m_connection_change_flag{ false };     //如果执行过IniConnection()函数，该flag会置为true\n    bool m_is_foreground_fullscreen{ false };   //指示前台窗口是否正在全局显示\n    bool m_menu_popuped{ false };               //指示当前是否有菜单处于弹出状态\n\n    HDC m_desktop_dc;\n\n    string m_connection_name_preferd{ theApp.m_cfg_data.m_connection_name };          //保存用户手动选择的网络连接名称\n\n    //CHighResolutionTimer m_timer;           // 采用多媒体定时器(也防止了界面阻塞出现的卡顿现象)\n    CCriticalSection m_critical;\n    static UINT MonitorThreadCallback(LPVOID dwUser);\n    bool m_is_monitor_thread_runing{ false };\n\n    CString GetMouseTipsInfo();     //获取鼠标提示信息\n    void SetTransparency();         //根据m_transparency的值设置窗口透明度\n    void SetTransparency(int transparency);\n    void SetAlwaysOnTop();          //根据m_always_on_top的值设置窗口置顶\n    void SetMousePenetrate();       //根据m_mouse_penetrate的值设置是否鼠标穿透\n    POINT CalculateWindowMoveOffset(CRect rect, bool screen_changed);  //计算当窗口处于屏幕区域外时，移动到屏幕区域需要移动的位置\n    void CheckWindowPos(bool screen_changed = false);          //测试窗口的位置，如窗口的位置在屏幕外，则移动窗口使其全部都在屏幕内，并返回新位置\n    void GetScreenSize();           //获取屏幕的大小\n\n    void AutoSelect();\n    //void UpdateConnections();\n    //自动选择连接\n    void IniConnection();   //初始化连接\n\n    MIB_IFROW GetConnectIfTable(int connection_index);    //获取当前选择的网络连接的MIB_IFROW对象。connection_index为m_connections中的索引\n    NetWorkConection GetConnection(int connection_index); //获取当前选择的网络连接的NetWorkConection对象。connection_index为m_connections中的索引\n\n    void IniConnectionMenu(CMenu* pMenu);   //初始化“选择网络连接”菜单\n    void IniTaskBarConnectionMenu();        //初始化任务栏窗口的“选择网络连接”菜单\n    void SetConnectionMenuState(CMenu* pMenu);      //设置“选择网络连接”菜单中选中的项目\n\n    void CloseTaskBarWnd(); //关闭任务栏窗口\n    void OpenTaskBarWnd();  //打开任务栏窗口\n\n    void AddNotifyIcon();       //添加通知区图标\n    void DeleteNotifyIcon();\n    void ShowNotifyTip(const wchar_t* title, const wchar_t* message);       //显示通知区提示\n    void UpdateNotifyIconTip();     //更新通知区图标的鼠标提示\n\n    void SaveHistoryTraffic();\n    void LoadHistoryTraffic();\n    void BackupHistoryTrafficFile();\n\n    void _OnOptions(int tab);   //打开“选项”对话框的处理，参数为打开时切换的标签\n\n    void ApplySettings(COptionsDlg& optionsDlg);\n\n    void SetItemPosition();     //设置显示的4个项目的位置\n    void LoadSkinLayout();      //从当前皮肤获取布局数据\n\n    void LoadBackGroundImage();\n    void SetTextFont();\n\n    bool IsTaskbarWndValid() const;\n\n    void TaskbarShowHideItem(DisplayItem type);\n\n    //判断一个点在哪个显示项目的区域内，并保存到m_clicked_item\n    void CheckClickedItem(CPoint point);\n\npublic:\n    //void ApplySettings();\n    bool IsTemperatureNeeded() const;       //判断是否需要显示温度信息\n\nprotected:\n    // 生成的消息映射函数\n    virtual BOOL OnInitDialog();\n    //  afx_msg void OnPaint();\n    afx_msg HCURSOR OnQueryDragIcon();\n    DECLARE_MESSAGE_MAP()\npublic:\n    //  afx_msg LRESULT OnNcHitTest(CPoint point);\n    afx_msg void OnTimer(UINT_PTR nIDEvent);\n    //  afx_msg void OnRButtonDown(UINT nFlags, CPoint point);\n    afx_msg void OnRButtonUp(UINT nFlags, CPoint point);\n    afx_msg void OnLButtonDown(UINT nFlags, CPoint point);\n    afx_msg void OnNetworkInfo();\n    afx_msg void OnAlwaysOnTop();\n    afx_msg void OnTransparency100();\n    afx_msg void OnTransparency80();\n    afx_msg void OnTransparency60();\n    afx_msg void OnTransparency40();\n    afx_msg void OnClose();\n    virtual BOOL OnCommand(WPARAM wParam, LPARAM lParam);\n    afx_msg void OnInitMenu(CMenu* pMenu);\n    virtual BOOL PreTranslateMessage(MSG* pMsg);\n    afx_msg void OnLockWindowPos();\n    afx_msg void OnMove(int x, int y);\nprotected:\n    afx_msg LRESULT OnNotifyIcon(WPARAM wParam, LPARAM lParam);\npublic:\n    afx_msg void OnShowNotifyIcon();\n    afx_msg void OnDestroy();\n    afx_msg void OnShowCpuMemory();\n    afx_msg void OnMousePenetrate();\n    //afx_msg void OnTextColor();\n    afx_msg void OnShowTaskBarWnd();\n    afx_msg void OnAppAbout();\n    afx_msg void OnShowCpuMemory2();\n    afx_msg void OnShowMainWnd();\n    afx_msg void OnChangeSkin();\n    afx_msg LRESULT OnTaskBarCreated(WPARAM wParam, LPARAM lParam);\n    //afx_msg void OnSetFont();\n    afx_msg void OnTrafficHistory();\n    afx_msg void OnMouseMove(UINT nFlags, CPoint point);\n    afx_msg void OnLButtonDblClk(UINT nFlags, CPoint point);\n    afx_msg void OnOptions();\n    afx_msg void OnOptions2();\nprotected:\n    afx_msg LRESULT OnExitmenuloop(WPARAM wParam, LPARAM lParam);\npublic:\n    afx_msg void OnChangeNotifyIcon();\n    afx_msg void OnAlowOutOfBorder();\n    afx_msg void OnCheckUpdate();\nprotected:\n    afx_msg LRESULT OnTaskbarMenuPopedUp(WPARAM wParam, LPARAM lParam);\npublic:\n    afx_msg void OnShowNetSpeed();\n    afx_msg BOOL OnQueryEndSession();\n    afx_msg void OnPaint();\nprotected:\n    afx_msg LRESULT OnDpichanged(WPARAM wParam, LPARAM lParam);\n    afx_msg LRESULT OnTaskbarWndClosed(WPARAM wParam, LPARAM lParam);\n    afx_msg LRESULT OnMonitorInfoUpdated(WPARAM wParam, LPARAM lParam);\n    afx_msg LRESULT OnDisplaychange(WPARAM wParam, LPARAM lParam);\n    afx_msg LRESULT OnReopenTaksbarWnd(WPARAM wParam, LPARAM lParam);\npublic:\n    afx_msg void OnExitSizeMove();\n    afx_msg void OnPluginManage();\n    afx_msg void OnOpenTaskManager();\nprotected:\n    afx_msg LRESULT OnSettingsApplied(WPARAM wParam, LPARAM lParam);\npublic:\n    afx_msg void OnDisplaySettings();\n    afx_msg void OnLButtonUp(UINT nFlags, CPoint point);\n    afx_msg void OnRefreshConnectionList();\n};\n"
  },
  {
    "path": "TrafficMonitor/UpdateHelper.cpp",
    "content": "﻿#include \"stdafx.h\"\n#include \"UpdateHelper.h\"\n#include \"SimpleXML.h\"\n#include \"Common.h\"\n\n\nCUpdateHelper::CUpdateHelper()\n{\n}\n\n\nCUpdateHelper::~CUpdateHelper()\n{\n}\n\nvoid CUpdateHelper::SetUpdateSource(UpdateSource update_source)\n{\n    m_update_source = update_source;\n}\n\nbool CUpdateHelper::CheckForUpdate()\n{\n    wstring version_info;\n    //使用GitHub更新源\n    if (m_update_source == UpdateSource::GitHubSource)\n    {\n        if (CCommon::GetURL(L\"https://raw.githubusercontent.com/zhongyang219/TrafficMonitor/master/version_utf8.info\", version_info, true))     //获取版本信息\n        {\n            m_row_data = true;\n        }\n        else if (CCommon::GetURL(L\"https://github.com/zhongyang219/TrafficMonitor/blob/master/version_utf8.info\", version_info, true))      //获取版本信息\n        {\n            m_row_data = false;\n        }\n        else\n        {\n            return false;\n        }\n\n        if (!m_row_data)\n        {\n            size_t index = version_info.find(L\"﻿&lt;version&gt;\");\n            if (index != std::wstring::npos)\n                version_info = version_info.substr(index);\n\n            CString str_version_info = version_info.c_str();\n            str_version_info.Replace(L\"&lt;\", L\"<\");\n            str_version_info.Replace(L\"&gt;\", L\">\");\n\n            version_info = str_version_info;\n        }\n    }\n    //使用Gitee更新源\n    else\n    {\n        if (!CCommon::GetURL(L\"https://gitee.com/zhongyang219/TrafficMonitor/raw/master/version_utf8.info\", version_info, true))     //获取版本信息\n            return false;\n    }\n\n    ParseUpdateInfo(version_info);\n\n    return true;\n}\n\nvoid CUpdateHelper::ParseUpdateInfo(wstring version_info)\n{\n    CSimpleXML version_xml;\n    version_xml.LoadXMLContentDirect(version_info);\n\n    m_version = version_xml.GetNode(L\"version\");\n    wstring str_source_tag = (m_update_source == UpdateSource::GitHubSource ? L\"GitHub\" : L\"Gitee\");\n    wstring str_link_tag, str_link_tag_x64;\n#ifdef WITHOUT_TEMPERATURE\n    str_link_tag = L\"link_without_temperature\";\n    str_link_tag_x64 = L\"link_without_temperature_x64\";\n#else\n    str_link_tag = L\"link\";\n    str_link_tag_x64 = L\"link_x64\";\n#endif\n    m_link64 = version_xml.GetNode(str_link_tag_x64.c_str(), str_source_tag.c_str());\n    m_link = version_xml.GetNode(str_link_tag.c_str(), str_source_tag.c_str());\n    CString contents_zh_cn = version_xml.GetNode(L\"contents_zh_cn\", L\"update_contents\").c_str();\n    CString contents_en = version_xml.GetNode(L\"contents_en\", L\"update_contents\").c_str();\n    CString contents_zh_tw = version_xml.GetNode(L\"contents_zh_tw\", L\"update_contents\").c_str();\n    contents_zh_cn.Replace(L\"\\\\n\", L\"\\r\\n\");\n    contents_en.Replace(L\"\\\\n\", L\"\\r\\n\");\n    contents_zh_tw.Replace(L\"\\\\n\", L\"\\r\\n\");\n    m_contents_zh_cn = contents_zh_cn;\n    m_contents_en = contents_en;\n    m_contents_zh_tw = contents_zh_tw;\n}\n\nconst std::wstring& CUpdateHelper::GetVersion() const\n{\n    return m_version;\n}\n\nconst std::wstring& CUpdateHelper::GetLink() const\n{\n    return m_link;\n}\n\nconst std::wstring& CUpdateHelper::GetLink64() const\n{\n    return m_link64;\n}\n\nconst std::wstring& CUpdateHelper::GetContentsEn() const\n{\n    return m_contents_en;\n}\n\nconst std::wstring& CUpdateHelper::GetContentsZhCn() const\n{\n    return m_contents_zh_cn;\n}\n\nconst std::wstring& CUpdateHelper::GetContentsZhTw() const\n{\n    return m_contents_zh_tw;\n}\n\nbool CUpdateHelper::IsRowData()\n{\n    return m_row_data;\n}\n"
  },
  {
    "path": "TrafficMonitor/UpdateHelper.h",
    "content": "﻿#pragma once\nclass CUpdateHelper\n{\npublic:\n    CUpdateHelper();\n    ~CUpdateHelper();\n\n    enum class UpdateSource\n    {\n        GitHubSource,\n        GiteeSource\n    };\n\n    void SetUpdateSource(UpdateSource update_source);\n\n    bool CheckForUpdate();\n\n    const std::wstring& GetVersion() const;\n    const std::wstring& GetLink() const;\n    const std::wstring& GetLink64() const;\n    const std::wstring& GetContentsEn() const;\n    const std::wstring& GetContentsZhCn() const;\n    const std::wstring& GetContentsZhTw() const;\n    bool IsRowData();\n\nprivate:\n    void ParseUpdateInfo(wstring version_info);\n\nprivate:\n    std::wstring m_version;\n    std::wstring m_link;\n    std::wstring m_link64;\n    std::wstring m_contents_en;\n    std::wstring m_contents_zh_cn;\n    std::wstring m_contents_zh_tw;\n    bool m_row_data{ true };\n    UpdateSource m_update_source{ UpdateSource::GitHubSource };\n};\n"
  },
  {
    "path": "TrafficMonitor/WIC.cpp",
    "content": "#include \"stdafx.h\"\n#include \"WIC.h\"\n\nCWICFactory CWICFactory::m_instance;\n\nCWICFactory::CWICFactory()\n{\n    //ʼm_pWICFactory\n    _hrOleInit = ::OleInitialize(NULL);\n#ifndef COMPILE_FOR_WINXP\n    CoCreateInstance(CLSID_WICImagingFactory, NULL, CLSCTX_INPROC_SERVER, IID_PPV_ARGS(&m_pWICFactory));\n    if (m_pWICFactory == nullptr)\n        CoCreateInstance(CLSID_WICImagingFactory1, NULL, CLSCTX_INPROC_SERVER, IID_PPV_ARGS(&m_pWICFactory));\n#endif\n}\n\nCWICFactory::~CWICFactory()\n{\n    if (m_pWICFactory)\n    {\n        m_pWICFactory->Release();\n        m_pWICFactory = NULL;\n    }\n\n    if (SUCCEEDED(_hrOleInit))\n    {\n        OleUninitialize();\n    }\n}\n\n\n//////////////////////////////////////////////////////////\n\ntypedef DWORD ARGB;\n\nCMenuIcon::CMenuIcon()\n{\n}\n\nCMenuIcon::~CMenuIcon()\n{\n}\n\nHRESULT CMenuIcon::AddIconToMenuItem(HMENU hmenu, int iMenuItem, BOOL fByPosition, HICON hicon)\n{\n#ifndef COMPILE_FOR_WINXP\n    if (CWICFactory::GetWIC() == nullptr)\n        return 0;\n    HBITMAP hbmp = NULL;\n\n    IWICBitmap *pBitmap;\n    HRESULT hr = CWICFactory::GetWIC()->CreateBitmapFromHICON(hicon, &pBitmap);\n    if (SUCCEEDED(hr))\n    {\n        IWICFormatConverter *pConverter;\n        hr = CWICFactory::GetWIC()->CreateFormatConverter(&pConverter);\n        if (SUCCEEDED(hr))\n        {\n            hr = pConverter->Initialize(pBitmap, GUID_WICPixelFormat32bppPBGRA, WICBitmapDitherTypeNone, NULL, 0.0f, WICBitmapPaletteTypeCustom);\n            if (SUCCEEDED(hr))\n            {\n                UINT cx, cy;\n                hr = pConverter->GetSize(&cx, &cy);\n                if (SUCCEEDED(hr))\n                {\n                    const SIZE sizIcon = { (int)cx, -(int)cy };\n                    BYTE *pbBuffer;\n                    hr = Create32BitHBITMAP(NULL, &sizIcon, reinterpret_cast<void **>(&pbBuffer), &hbmp);\n                    if (SUCCEEDED(hr))\n                    {\n                        const UINT cbStride = cx * sizeof(ARGB);\n                        const UINT cbBuffer = cy * cbStride;\n                        hr = pConverter->CopyPixels(NULL, cbStride, cbBuffer, pbBuffer);\n                    }\n                }\n            }\n\n            pConverter->Release();\n        }\n\n        pBitmap->Release();\n    }\n\n    if (SUCCEEDED(hr))\n    {\n        hr = AddBitmapToMenuItem(hmenu, iMenuItem, fByPosition, hbmp);\n    }\n\n    if (FAILED(hr))\n    {\n        DeleteObject(hbmp);\n        hbmp = NULL;\n    }\n\n    return hr;\n\n#else\n    return 0;\n#endif\n}\n\nHRESULT CMenuIcon::AddBitmapToMenuItem(HMENU hmenu, int iItem, BOOL fByPosition, HBITMAP hbmp)\n{\n    HRESULT hr = E_FAIL;\n\n    MENUITEMINFO mii = { sizeof(mii) };\n    mii.fMask = MIIM_BITMAP;\n    mii.hbmpItem = hbmp;\n    if (SetMenuItemInfo(hmenu, iItem, fByPosition, &mii))\n    {\n        hr = S_OK;\n    }\n\n    return hr;\n}\n\nvoid CMenuIcon::InitBitmapInfo(BITMAPINFO * pbmi, ULONG cbInfo, LONG cx, LONG cy, WORD bpp)\n{\n    ZeroMemory(pbmi, cbInfo);\n    pbmi->bmiHeader.biSize = sizeof(BITMAPINFOHEADER);\n    pbmi->bmiHeader.biPlanes = 1;\n    pbmi->bmiHeader.biCompression = BI_RGB;\n\n    pbmi->bmiHeader.biWidth = cx;\n    pbmi->bmiHeader.biHeight = cy;\n    pbmi->bmiHeader.biBitCount = bpp;\n}\n\nHRESULT CMenuIcon::Create32BitHBITMAP(HDC hdc, const SIZE * psize, void ** ppvBits, HBITMAP * phBmp)\n{\n    *phBmp = NULL;\n\n    BITMAPINFO bmi;\n    InitBitmapInfo(&bmi, sizeof(bmi), psize->cx, psize->cy, 32);\n\n    HDC hdcUsed = hdc ? hdc : GetDC(NULL);\n    if (hdcUsed)\n    {\n        *phBmp = CreateDIBSection(hdcUsed, &bmi, DIB_RGB_COLORS, ppvBits, NULL, 0);\n        if (hdc != hdcUsed)\n        {\n            ReleaseDC(NULL, hdcUsed);\n        }\n    }\n    return (NULL == *phBmp) ? E_OUTOFMEMORY : S_OK;\n}\n"
  },
  {
    "path": "TrafficMonitor/WIC.h",
    "content": "//ʹ Windows ӳWICͼ꣨HICONתΪ͸ȵPARGB32λͼӵ˵\n//https://docs.microsoft.com/en-us/previous-versions/bb757020(v=msdn.10)\n\n#pragma once\nclass CWICFactory\n{\npublic:\n    ~CWICFactory();\n    static IWICImagingFactory* GetWIC() { return m_instance.m_pWICFactory; }\n\nprivate:\n    HRESULT _hrOleInit{};\n    IWICImagingFactory *m_pWICFactory{};\n\n    static CWICFactory m_instance;      //CWICFactoryΨһĶ\n\nprivate:\n    CWICFactory();\n};\n\nclass CMenuIcon\n{\npublic:\n    CMenuIcon();\n    ~CMenuIcon();\n\n    //һ˵ͼ\n    static HRESULT AddIconToMenuItem(HMENU hmenu, int iMenuItem, BOOL fByPosition, HICON hicon);\n\nprivate:\n    static HRESULT AddBitmapToMenuItem(HMENU hmenu, int iItem, BOOL fByPosition, HBITMAP hbmp);\n    static void InitBitmapInfo(__out_bcount(cbInfo) BITMAPINFO *pbmi, ULONG cbInfo, LONG cx, LONG cy, WORD bpp);\n    static HRESULT Create32BitHBITMAP(HDC hdc, const SIZE *psize, __deref_opt_out void **ppvBits, __out HBITMAP* phBmp);\n};\n\n"
  },
  {
    "path": "TrafficMonitor/WinVersionHelper.cpp",
    "content": "﻿#include \"stdafx.h\"\n#include \"WinVersionHelper.h\"\n#include \"WindowsSettingHelper.h\"\n\n\nCWinVersionHelper::CWinVersionHelper()\n{\n\tDWORD dwMajorVer{}, dwMinorVer{}, dwBuildNumber{};\n\tHMODULE hModNtdll{};\n\tif (hModNtdll = ::LoadLibraryW(L\"ntdll.dll\"))\n\t{\n\t\ttypedef void (WINAPI *pfRTLGETNTVERSIONNUMBERS)(DWORD*, DWORD*, DWORD*);\n\t\tpfRTLGETNTVERSIONNUMBERS pfRtlGetNtVersionNumbers;\n\t\tpfRtlGetNtVersionNumbers = (pfRTLGETNTVERSIONNUMBERS)::GetProcAddress(hModNtdll, \"RtlGetNtVersionNumbers\");\n\t\tif (pfRtlGetNtVersionNumbers)\n\t\t{\n\t\t\tpfRtlGetNtVersionNumbers(&dwMajorVer, &dwMinorVer, &dwBuildNumber);\n\t\t\tdwBuildNumber &= 0x0ffff;\n\t\t}\n\t\t::FreeLibrary(hModNtdll);\n\t\thModNtdll = NULL;\n\t}\n\tm_major_version = dwMajorVer;\n\tm_minor_version = dwMinorVer;\n\tm_build_number = dwBuildNumber;\n\n    CWindowsSettingHelper::CheckWindows10LightTheme();\n}\n\n\nCWinVersionHelper::~CWinVersionHelper()\n{\n}\n\nbool CWinVersionHelper::IsWindows11OrLater() const\n{\n\tif (m_major_version > 10)\n\t\treturn true;\n\telse if (m_major_version == 10 && m_minor_version > 0)\n\t\treturn true;\n\telse if (m_major_version == 10 && m_minor_version == 0 && m_build_number >= 21996)\n\t\treturn true;\n\telse return false;\n}\n\nbool CWinVersionHelper::IsWindows10FallCreatorOrLater() const\n{\n\tif (m_major_version > 10)\n\t\treturn true;\n\telse if (m_major_version == 10 && m_minor_version > 0)\n\t\treturn true;\n\telse if (m_major_version == 10 && m_minor_version == 0 && m_build_number >= 16299)\n\t\treturn true;\n\telse return false;\n}\n\nbool CWinVersionHelper::IsWindows7() const\n{\n\treturn (m_major_version == 6 && m_minor_version == 1);\n}\n\nbool CWinVersionHelper::IsWindows8Or8point1() const\n{\n\treturn (m_major_version == 6 && m_minor_version > 1);\n}\n\nbool CWinVersionHelper::IsWindows8OrLater() const\n{\n    if (m_major_version > 6)\n        return true;\n    else if (m_major_version == 6 && m_minor_version >= 2)\n        return true;\n    else return false;\n}\n\nbool CWinVersionHelper::IsWindows10OrLater() const\n{\n    return m_major_version >= 10;\n}\n"
  },
  {
    "path": "TrafficMonitor/WinVersionHelper.h",
    "content": "﻿#pragma once\nclass CWinVersionHelper\n{\npublic:\n\tCWinVersionHelper();\n\t~CWinVersionHelper();\n\n\tbool IsWindows11OrLater() const;\t\t\t//判断当前Windows版本是否为Win11或更新的版本\n\tbool IsWindows10FallCreatorOrLater() const;\t\t//判断当前Windows版本是否为Win10秋季创意者更新或更新的版本\n\tbool IsWindows7() const;\t\t\t\t\t//判断Windows版本是否为Windows7\n\tbool IsWindows8Or8point1() const;\t\t\t//判断Windows版本是否为Windows8或Windows8.1\n    bool IsWindows8OrLater() const;\n    bool IsWindows10OrLater() const;\n\n\tint GetMajorVersion() const { return m_major_version; }\n\tint GetMinorVersion() const { return m_minor_version; }\n\tint GetBuildNumber() const { return m_build_number; }\nprotected:\n\tint m_major_version{};\n\tint m_minor_version{};\n\tint m_build_number{};\n};\n"
  },
  {
    "path": "TrafficMonitor/WindowsSettingHelper.cpp",
    "content": "#include \"stdafx.h\"\n#include \"WindowsSettingHelper.h\"\n#include \"TrafficMonitor.h\"\n\nbool CWindowsSettingHelper::m_light_theme{};\nbool CWindowsSettingHelper::IsWindows10LightTheme()\n{\n    return m_light_theme;\n}\n\nvoid CWindowsSettingHelper::CheckWindows10LightTheme()\n{\n    if (theApp.m_win_version.IsWindows10OrLater())\n    {\n        HKEY hKey;\n        DWORD dwThemeData(0);\n        LONG lRes = RegOpenKeyExW(HKEY_CURRENT_USER, L\"Software\\\\Microsoft\\\\Windows\\\\CurrentVersion\\\\Themes\\\\Personalize\", 0, KEY_READ, &hKey);\n        if (lRes == ERROR_SUCCESS) {\n            GetDWORDRegKeyData(hKey, L\"SystemUsesLightTheme\", dwThemeData);\n            m_light_theme = (dwThemeData != 0);\n        }\n        else\n        {\n            m_light_theme = false;\n        }\n        RegCloseKey(hKey);\n    }\n    else\n    {\n        m_light_theme = false;\n    }\n}\n\nbool CWindowsSettingHelper::IsDotNetFramework4Point5Installed()\n{\n    DWORD netFramewordRelease{};\n    if (!GetDWORDRegKeyData(HKEY_LOCAL_MACHINE, L\"Software\\\\Microsoft\\\\NET Framework Setup\\\\NDP\\\\v4\\\\Full\", L\"Release\", netFramewordRelease))\n        return false;\n    return netFramewordRelease >= 379893;\n}\n\nbool CWindowsSettingHelper::IsTaskbarSearchBtnShown()\n{\n    DWORD data{};\n    if (!GetDWORDRegKeyData(HKEY_CURRENT_USER, L\"Software\\\\Microsoft\\\\Windows\\\\CurrentVersion\\\\Search\", L\"SearchboxTaskbarMode\", data))\n        return false;\n    return data != 0;\n}\n\nbool CWindowsSettingHelper::IsTaskbarTaskViewBtnShown()\n{\n    DWORD data{};\n    if (!GetDWORDRegKeyData(HKEY_CURRENT_USER, L\"Software\\\\Microsoft\\\\Windows\\\\CurrentVersion\\\\Explorer\\\\Advanced\", L\"ShowTaskViewButton\", data))\n        return false;\n    return data != 0;\n}\n\nbool CWindowsSettingHelper::IsTaskbarWidgetsBtnShown()\n{\n    DWORD data{};\n    if (!GetDWORDRegKeyData(HKEY_CURRENT_USER, L\"Software\\\\Microsoft\\\\Windows\\\\CurrentVersion\\\\Explorer\\\\Advanced\", L\"TaskbarDa\", data))\n        return false;\n    return data != 0;\n}\n\nbool CWindowsSettingHelper::IsTaskbarChartBtnShown()\n{\n    DWORD data{};\n    if (!GetDWORDRegKeyData(HKEY_CURRENT_USER, L\"Software\\\\Microsoft\\\\Windows\\\\CurrentVersion\\\\Explorer\\\\Advanced\", L\"TaskbarMn\", data))\n        return false;\n    return data != 0;\n}\n\nbool CWindowsSettingHelper::IsTaskbarCenterAlign()\n{\n    DWORD data{};\n    if (!GetDWORDRegKeyData(HKEY_CURRENT_USER, L\"Software\\\\Microsoft\\\\Windows\\\\CurrentVersion\\\\Explorer\\\\Advanced\", L\"TaskbarAl\", data))\n        return false;\n    return data != 0;\n}\n\nLONG CWindowsSettingHelper::GetDWORDRegKeyData(HKEY hKey, const wstring& strValueName, DWORD& dwValueData)\n{\n    DWORD dwBufferSize(sizeof(DWORD));\n    DWORD dwResult(0);\n    LONG lError = ::RegQueryValueExW(hKey, strValueName.c_str(), NULL, NULL, reinterpret_cast<LPBYTE>(&dwResult), &dwBufferSize);\n    if (lError == ERROR_SUCCESS)\n        dwValueData = dwResult;\n    return lError;\n}\n\nbool CWindowsSettingHelper::GetDWORDRegKeyData(HKEY keyParent, const wstring& strKeyName, const wstring& strValueName, DWORD& dwValueData)\n{\n    CRegKey key;\n    if (key.Open(keyParent, strKeyName.c_str(), KEY_READ) != ERROR_SUCCESS)\n        return false;\n    return (key.QueryDWORDValue(strValueName.c_str(), dwValueData) == ERROR_SUCCESS);\n}\n"
  },
  {
    "path": "TrafficMonitor/WindowsSettingHelper.h",
    "content": "#pragma once\nclass CWindowsSettingHelper\n{\npublic:\n    static bool IsWindows10LightTheme();\t\t\t\t//жWindowsǷΪǳɫ\n    static void CheckWindows10LightTheme();\n    static bool IsDotNetFramework4Point5Installed();\t//жǷװ.Net Framework 4.5 (https://docs.microsoft.com/zh-cn/dotnet/framework/migration-guide/how-to-determine-which-versions-are-installed)\n\n    //Windows11\n    static bool IsTaskbarSearchBtnShown();              //Ƿʾˡť\n    static bool IsTaskbarTaskViewBtnShown();            //Ƿʾˡͼť\n    static bool IsTaskbarWidgetsBtnShown();             //ǷʾˡСť\n    static bool IsTaskbarChartBtnShown();               //Ƿʾˡ족ť\n    static bool IsTaskbarCenterAlign();                  //Ƿ\n\nprivate:\n    static LONG GetDWORDRegKeyData(HKEY hKey, const wstring& strValueName, DWORD& dwValueData);\n    static bool GetDWORDRegKeyData(HKEY keyParent, const wstring& strKeyName, const wstring& strValueName, DWORD& dwValueData);\n\nprivate:\n    static \tbool m_light_theme;\n};\n\n"
  },
  {
    "path": "TrafficMonitor/auto_start_helper.cpp",
    "content": "﻿#include \"stdafx.h\"\n#include \"auto_start_helper.h\"\n\n#include <Lmcons.h>\n\n#include <comdef.h>\n#include <taskschd.h>\n#include \"SimpleXML.h\"\n# pragma comment(lib, \"taskschd.lib\")\n\n// Helper macros from wix.\n// TODO: use \"s\" and \"...\" parameters to report errors from these functions.\n#define ExitOnFailure(x, s, ...) \\\n    if (FAILED(x))               \\\n    {                            \\\n        goto LExit;              \\\n    }\n#define ExitWithLastError(x, s, ...)       \\\n    {                                      \\\n        DWORD Dutil_er = ::GetLastError(); \\\n        x = HRESULT_FROM_WIN32(Dutil_er);  \\\n        if (!FAILED(x))                    \\\n        {                                  \\\n            x = E_FAIL;                    \\\n        }                                  \\\n        goto LExit;                        \\\n    }\n#define ExitFunction() \\\n    {                  \\\n        goto LExit;    \\\n    }\n\nconst DWORD USERNAME_DOMAIN_LEN = DNLEN + UNLEN + 2; // Domain Name + '\\' + User Name + '\\0'\nconst DWORD USERNAME_LEN = UNLEN + 1; // User Name + '\\0'\n\nbool create_auto_start_task_for_this_user(bool runElevated)\n{\n    HRESULT hr = S_OK;\n\n    WCHAR username_domain[USERNAME_DOMAIN_LEN];\n    WCHAR username[USERNAME_LEN];\n\n    std::wstring wstrTaskName;\n\n    ITaskService* pService = NULL;\n    ITaskFolder* pTaskFolder = NULL;\n    ITaskDefinition* pTask = NULL;\n    IRegistrationInfo* pRegInfo = NULL;\n    ITaskSettings* pSettings = NULL;\n    ITriggerCollection* pTriggerCollection = NULL;\n    IRegisteredTask* pRegisteredTask = NULL;\n\n    // ------------------------------------------------------\n    // Get the Domain/Username for the trigger.\n    if (!GetEnvironmentVariable(L\"USERNAME\", username, USERNAME_LEN))\n    {\n        ExitWithLastError(hr, \"Getting username failed: %x\", hr);\n    }\n    if (!GetEnvironmentVariable(L\"USERDOMAIN\", username_domain, USERNAME_DOMAIN_LEN))\n    {\n        ExitWithLastError(hr, \"Getting the user's domain failed: %x\", hr);\n    }\n    wcscat_s(username_domain, L\"\\\\\");\n    wcscat_s(username_domain, username);\n\n    // Task Name.\n    wstrTaskName = L\"Autorun for \";\n    wstrTaskName += username;\n\n    // Get the executable path passed to the custom action.\n    WCHAR wszExecutablePath[MAX_PATH];\n    GetModuleFileName(NULL, wszExecutablePath, MAX_PATH);\n\n    // ------------------------------------------------------\n    // Create an instance of the Task Service.\n    hr = CoCreateInstance(CLSID_TaskScheduler,\n                          NULL,\n                          CLSCTX_INPROC_SERVER,\n                          IID_ITaskService,\n                          (void**)&pService);\n    ExitOnFailure(hr, \"Failed to create an instance of ITaskService: %x\", hr);\n\n    // Connect to the task service.\n    hr = pService->Connect(_variant_t(), _variant_t(), _variant_t(), _variant_t());\n    ExitOnFailure(hr, \"ITaskService::Connect failed: %x\", hr);\n\n    // ------------------------------------------------------\n    // Get the TrafficMonitor task folder. Creates it if it doesn't exist.\n    hr = pService->GetFolder(_bstr_t(L\"\\\\TrafficMonitor\"), &pTaskFolder);\n    if (FAILED(hr))\n    {\n        // Folder doesn't exist. Get the Root folder and create the TrafficMonitor subfolder.\n        ITaskFolder* pRootFolder = NULL;\n        hr = pService->GetFolder(_bstr_t(L\"\\\\\"), &pRootFolder);\n        ExitOnFailure(hr, \"Cannot get Root Folder pointer: %x\", hr);\n        hr = pRootFolder->CreateFolder(_bstr_t(L\"\\\\TrafficMonitor\"), _variant_t(L\"\"), &pTaskFolder);\n        if (FAILED(hr))\n        {\n            pRootFolder->Release();\n            ExitOnFailure(hr, \"Cannot create TrafficMonitor task folder: %x\", hr);\n        }\n    }\n\n    // If the task exists, just enable it.\n    {\n        IRegisteredTask* pExistingRegisteredTask = NULL;\n        hr = pTaskFolder->GetTask(_bstr_t(wstrTaskName.c_str()), &pExistingRegisteredTask);\n        if (SUCCEEDED(hr))\n        {\n            // Task exists, try enabling it.\n            hr = pExistingRegisteredTask->put_Enabled(VARIANT_TRUE);\n            pExistingRegisteredTask->Release();\n            if (SUCCEEDED(hr))\n            {\n                // Function enable. Sounds like a success.\n                ExitFunction();\n            }\n        }\n    }\n\n    // Create the task builder object to create the task.\n    hr = pService->NewTask(0, &pTask);\n    ExitOnFailure(hr, \"Failed to create a task definition: %x\", hr);\n\n    // ------------------------------------------------------\n    // Get the registration info for setting the identification.\n    hr = pTask->get_RegistrationInfo(&pRegInfo);\n    ExitOnFailure(hr, \"Cannot get identification pointer: %x\", hr);\n    hr = pRegInfo->put_Author(_bstr_t(username_domain));\n    ExitOnFailure(hr, \"Cannot put identification info: %x\", hr);\n\n    // ------------------------------------------------------\n    // Create the settings for the task\n    hr = pTask->get_Settings(&pSettings);\n    ExitOnFailure(hr, \"Cannot get settings pointer: %x\", hr);\n\n    hr = pSettings->put_StartWhenAvailable(VARIANT_FALSE);\n    ExitOnFailure(hr, \"Cannot put_StartWhenAvailable setting info: %x\", hr);\n    hr = pSettings->put_StopIfGoingOnBatteries(VARIANT_FALSE);\n    ExitOnFailure(hr, \"Cannot put_StopIfGoingOnBatteries setting info: %x\", hr);\n    hr = pSettings->put_ExecutionTimeLimit(_bstr_t(L\"PT0S\")); //Unlimited\n    ExitOnFailure(hr, \"Cannot put_ExecutionTimeLimit setting info: %x\", hr);\n    hr = pSettings->put_DisallowStartIfOnBatteries(VARIANT_FALSE);\n    ExitOnFailure(hr, \"Cannot put_DisallowStartIfOnBatteries setting info: %x\", hr);\n\n    // ------------------------------------------------------\n    // Get the trigger collection to insert the logon trigger.\n    hr = pTask->get_Triggers(&pTriggerCollection);\n    ExitOnFailure(hr, \"Cannot get trigger collection: %x\", hr);\n\n    // Add the logon trigger to the task.\n    {\n        ITrigger* pTrigger = NULL;\n        ILogonTrigger* pLogonTrigger = NULL;\n        hr = pTriggerCollection->Create(TASK_TRIGGER_LOGON, &pTrigger);\n        ExitOnFailure(hr, \"Cannot create the trigger: %x\", hr);\n\n        hr = pTrigger->QueryInterface(\n            IID_ILogonTrigger, (void**)&pLogonTrigger);\n        pTrigger->Release();\n        ExitOnFailure(hr, \"QueryInterface call failed for ILogonTrigger: %x\", hr);\n\n        hr = pLogonTrigger->put_Id(_bstr_t(L\"Trigger1\"));\n\n        // Timing issues may make explorer not be started when the task runs.\n        // Add a little delay to mitigate this.\n        hr = pLogonTrigger->put_Delay(_bstr_t(L\"PT03S\"));\n\n        // Define the user. The task will execute when the user logs on.\n        // The specified user must be a user on this computer.\n        hr = pLogonTrigger->put_UserId(_bstr_t(username_domain));\n        pLogonTrigger->Release();\n        ExitOnFailure(hr, \"Cannot add user ID to logon trigger: %x\", hr);\n    }\n\n    // ------------------------------------------------------\n    // Add an Action to the task. This task will execute the path passed to this custom action.\n    {\n        IActionCollection* pActionCollection = NULL;\n        IAction* pAction = NULL;\n        IExecAction* pExecAction = NULL;\n\n        // Get the task action collection pointer.\n        hr = pTask->get_Actions(&pActionCollection);\n        ExitOnFailure(hr, \"Cannot get Task collection pointer: %x\", hr);\n\n        // Create the action, specifying that it is an executable action.\n        hr = pActionCollection->Create(TASK_ACTION_EXEC, &pAction);\n        pActionCollection->Release();\n        ExitOnFailure(hr, \"Cannot create the action: %x\", hr);\n\n        // QI for the executable task pointer.\n        hr = pAction->QueryInterface(\n            IID_IExecAction, (void**)&pExecAction);\n        pAction->Release();\n        ExitOnFailure(hr, \"QueryInterface call failed for IExecAction: %x\", hr);\n\n        // Set the path of the executable to TrafficMonitor (passed as CustomActionData).\n        hr = pExecAction->put_Path(_bstr_t(wszExecutablePath));\n        pExecAction->Release();\n        ExitOnFailure(hr, \"Cannot set path of executable: %x\", hr);\n    }\n\n    // ------------------------------------------------------\n    // Create the principal for the task\n    {\n        IPrincipal* pPrincipal = NULL;\n        hr = pTask->get_Principal(&pPrincipal);\n        ExitOnFailure(hr, \"Cannot get principal pointer: %x\", hr);\n\n        // Set up principal information:\n        hr = pPrincipal->put_Id(_bstr_t(L\"Principal1\"));\n\n        hr = pPrincipal->put_UserId(_bstr_t(username_domain));\n\n        hr = pPrincipal->put_LogonType(TASK_LOGON_INTERACTIVE_TOKEN);\n\n        if (runElevated)\n        {\n            hr = pPrincipal->put_RunLevel(_TASK_RUNLEVEL::TASK_RUNLEVEL_HIGHEST);\n        }\n        else\n        {\n            hr = pPrincipal->put_RunLevel(_TASK_RUNLEVEL::TASK_RUNLEVEL_LUA);\n        }\n        pPrincipal->Release();\n        ExitOnFailure(hr, \"Cannot put principal run level: %x\", hr);\n    }\n    // ------------------------------------------------------\n    //  Save the task in the TrafficMonitor folder.\n    {\n        _variant_t SDDL_FULL_ACCESS_FOR_EVERYONE = L\"D:(A;;FA;;;WD)\";\n        hr = pTaskFolder->RegisterTaskDefinition(\n            _bstr_t(wstrTaskName.c_str()),\n            pTask,\n            TASK_CREATE_OR_UPDATE,\n            _variant_t(username_domain),\n            _variant_t(),\n            TASK_LOGON_INTERACTIVE_TOKEN,\n            SDDL_FULL_ACCESS_FOR_EVERYONE,\n            &pRegisteredTask);\n        ExitOnFailure(hr, \"Error saving the Task : %x\", hr);\n    }\n\nLExit:\n    if (pService)\n        pService->Release();\n    if (pTaskFolder)\n        pTaskFolder->Release();\n    if (pTask)\n        pTask->Release();\n    if (pRegInfo)\n        pRegInfo->Release();\n    if (pSettings)\n        pSettings->Release();\n    if (pTriggerCollection)\n        pTriggerCollection->Release();\n    if (pRegisteredTask)\n        pRegisteredTask->Release();\n\n    return (SUCCEEDED(hr));\n}\n\nbool delete_auto_start_task_for_this_user()\n{\n    HRESULT hr = S_OK;\n\n    WCHAR username[USERNAME_LEN];\n    std::wstring wstrTaskName;\n\n    ITaskService* pService = NULL;\n    ITaskFolder* pTaskFolder = NULL;\n\n    // ------------------------------------------------------\n    // Get the Username for the task.\n    if (!GetEnvironmentVariable(L\"USERNAME\", username, USERNAME_LEN))\n    {\n        ExitWithLastError(hr, \"Getting username failed: %x\", hr);\n    }\n\n    // Task Name.\n    wstrTaskName = L\"Autorun for \";\n    wstrTaskName += username;\n\n    // ------------------------------------------------------\n    // Create an instance of the Task Service.\n    hr = CoCreateInstance(CLSID_TaskScheduler,\n                          NULL,\n                          CLSCTX_INPROC_SERVER,\n                          IID_ITaskService,\n                          (void**)&pService);\n    ExitOnFailure(hr, \"Failed to create an instance of ITaskService: %x\", hr);\n\n    // Connect to the task service.\n    hr = pService->Connect(_variant_t(), _variant_t(), _variant_t(), _variant_t());\n    ExitOnFailure(hr, \"ITaskService::Connect failed: %x\", hr);\n\n    // ------------------------------------------------------\n    // Get the TrafficMonitor task folder.\n    hr = pService->GetFolder(_bstr_t(L\"\\\\TrafficMonitor\"), &pTaskFolder);\n    if (FAILED(hr))\n    {\n        // Folder doesn't exist. No need to disable a non-existing task.\n        hr = S_OK;\n        ExitFunction();\n    }\n\n    // ------------------------------------------------------\n    // If the task exists, disable.\n    {\n        IRegisteredTask* pExistingRegisteredTask = NULL;\n        hr = pTaskFolder->GetTask(_bstr_t(wstrTaskName.c_str()), &pExistingRegisteredTask);\n        if (SUCCEEDED(hr))\n        {\n            // Task exists, try disabling it.\n            hr = pTaskFolder->DeleteTask(_bstr_t(wstrTaskName.c_str()), 0);\n        }\n    }\n\nLExit:\n    if (pService)\n        pService->Release();\n    if (pTaskFolder)\n        pTaskFolder->Release();\n\n    return (SUCCEEDED(hr));\n}\n\nbool is_auto_start_task_active_for_this_user(std::wstring* path)\n{\n    HRESULT hr = S_OK;\n\n    WCHAR username[USERNAME_LEN];\n    std::wstring wstrTaskName;\n\n    ITaskService* pService = NULL;\n    ITaskFolder* pTaskFolder = NULL;\n\n    // ------------------------------------------------------\n    // Get the Username for the task.\n    if (!GetEnvironmentVariable(L\"USERNAME\", username, USERNAME_LEN))\n    {\n        ExitWithLastError(hr, \"Getting username failed: %x\", hr);\n    }\n\n    // Task Name.\n    wstrTaskName = L\"Autorun for \";\n    wstrTaskName += username;\n\n    // ------------------------------------------------------\n    // Create an instance of the Task Service.\n    hr = CoCreateInstance(CLSID_TaskScheduler,\n                          NULL,\n                          CLSCTX_INPROC_SERVER,\n                          IID_ITaskService,\n                          (void**)&pService);\n    ExitOnFailure(hr, \"Failed to create an instance of ITaskService: %x\", hr);\n\n    // Connect to the task service.\n    hr = pService->Connect(_variant_t(), _variant_t(), _variant_t(), _variant_t());\n    ExitOnFailure(hr, \"ITaskService::Connect failed: %x\", hr);\n\n    // ------------------------------------------------------\n    // Get the TrafficMonitor task folder.\n    hr = pService->GetFolder(_bstr_t(L\"\\\\TrafficMonitor\"), &pTaskFolder);\n    ExitOnFailure(hr, \"ITaskFolder doesn't exist: %x\", hr);\n\n    bool command_path_match{};\n    // ------------------------------------------------------\n    // If the task exists, disable.\n    {\n        IRegisteredTask* pExistingRegisteredTask = NULL;\n        hr = pTaskFolder->GetTask(_bstr_t(wstrTaskName.c_str()), &pExistingRegisteredTask);\n        if (SUCCEEDED(hr))\n        {\n            // Task exists, get its value.\n            VARIANT_BOOL is_enabled;\n            hr = pExistingRegisteredTask->get_Enabled(&is_enabled);\n            //判断已存在的任务计划命令的exe文件路径是否为当前exe的路径\n            BSTR xml_buff{};\n            pExistingRegisteredTask->get_Xml(&xml_buff);\n            CSimpleXML xml;\n            xml.LoadXMLContentDirect(xml_buff);\n            std::wstring command_path = xml.GetNode(L\"Command\", L\"Exec\");\n            if (path != nullptr)\n                *path = command_path;\n            WCHAR wszExecutablePath[MAX_PATH];\n            GetModuleFileName(NULL, wszExecutablePath, MAX_PATH);\n            command_path_match = (command_path == wszExecutablePath);\n\n            pExistingRegisteredTask->Release();\n            if (SUCCEEDED(hr))\n            {\n                // Got the value. Return it.\n                hr = (is_enabled == VARIANT_TRUE) ? S_OK : E_FAIL; // Fake success or fail to return the value.\n                ExitFunction();\n            }\n        }\n    }\n\nLExit:\n    if (pService)\n        pService->Release();\n    if (pTaskFolder)\n        pTaskFolder->Release();\n    //delete[] buff;\n\n    return (SUCCEEDED(hr) && command_path_match);\n}\n"
  },
  {
    "path": "TrafficMonitor/auto_start_helper.h",
    "content": "#pragma once\n\nbool is_auto_start_task_active_for_this_user(std::wstring* path);\nbool create_auto_start_task_for_this_user(bool runElevated);\nbool delete_auto_start_task_for_this_user();\n"
  },
  {
    "path": "TrafficMonitor/crashtool.cpp",
    "content": "﻿#include \"StdAfx.h\"\n#include \"crashtool.h\"\n\n#include <strsafe.h>\n#include <DbgHelp.h>\n#include <tchar.h>\n#include \"MessageDlg.h\"\n#include \"Common.h\"\n#include \"TrafficMonitor.h\"\n\n#pragma comment(lib, \"Dbghelp.lib\")\n\nclass CCrashReport\n{\npublic:\n    CCrashReport()\n    {\n        GetAppPath();\n    }\n    ~CCrashReport() {}\npublic:\n    // 生成MiniDump文件\n    void CreateMiniDump(EXCEPTION_POINTERS* pEP)\n    {\n        SYSTEMTIME stLocalTime;\n        ::GetLocalTime(&stLocalTime);\n        TCHAR szDumpFile[MAX_PATH] = {0};\n        ::StringCchPrintf(szDumpFile, _countof(szDumpFile), TEXT(\"%s%04d%02d%02d%02d%02d%02d_%s.dmp\"), m_szDumpFilePath,\n                          stLocalTime.wYear, stLocalTime.wMonth, stLocalTime.wDay, stLocalTime.wHour, stLocalTime.wMinute, stLocalTime.wSecond, m_szModuleFileName);\n\n        HANDLE hDumpFile;\n        hDumpFile = ::CreateFile(szDumpFile, GENERIC_READ | GENERIC_WRITE, FILE_SHARE_WRITE | FILE_SHARE_READ, 0, CREATE_ALWAYS, 0, 0);\n        if (INVALID_HANDLE_VALUE == hDumpFile)\n        {\n            return;\n        }\n\n\t\tm_dumpFile = szDumpFile;\n\n        MINIDUMP_EXCEPTION_INFORMATION ExpParam;\n        ExpParam.ThreadId = ::GetCurrentThreadId();\n        ExpParam.ExceptionPointers = pEP;\n        ExpParam.ClientPointers = TRUE;\n\n        // 生成minidump文件\n        BOOL bResult = ::MiniDumpWriteDump(GetCurrentProcess(), GetCurrentProcessId(), hDumpFile, MiniDumpNormal, &ExpParam, NULL, NULL);\n        ::CloseHandle(hDumpFile);\n    }\n\n\tvoid ShowCrashInfo()\n\t{\n\t\tCMessageDlg dlg;\n\t\tdlg.SetWindowTitle(APP_NAME);\n\t\tdlg.SetInfoText(CCommon::LoadText(IDS_ERROR_MESSAGE));\n\n\t\tCString info = CCommon::LoadTextFormat(IDS_CRASH_INFO, { m_dumpFile });\n\t\tinfo += _T(\"\\r\\n\");\n\t\tinfo += theApp.GetSystemInfoString();\n\t\tdlg.SetMessageText(info);\n\n        //设置图标\n        HICON hIcon;\n        HRESULT hr = LoadIconWithScaleDown(NULL, IDI_ERROR, theApp.DPI(32), theApp.DPI(32), &hIcon);\n\n        if (SUCCEEDED(hr))\n            dlg.SetMessageIcon(hIcon);\n\n        dlg.DoModal();\n\t}\n\nprivate:\n    void GetAppPath()\n    {\n        ZeroMemory(m_szModuleFileName, MAX_PATH);\n        ::GetModuleFileName(NULL, m_szDumpFilePath, _countof(m_szDumpFilePath));\n        for (int nIndex = (int)_tcslen(m_szDumpFilePath); nIndex >= 0; --nIndex)\n        {\n            if (m_szDumpFilePath[nIndex] == TEXT('\\\\'))\n            {\n                ::memmove(m_szModuleFileName, m_szDumpFilePath + nIndex + 1, (int)_tcslen(m_szDumpFilePath));\n                m_szDumpFilePath[nIndex + 1] = 0;\n                break;\n            }\n        }\n        ZeroMemory(m_szDumpFilePath, MAX_PATH);\n        if (!::GetTempPath(MAX_PATH, m_szDumpFilePath))\n        {\n            m_szDumpFilePath[0] = _T('C');\n            m_szDumpFilePath[1] = _T(':');\n            m_szDumpFilePath[2] = _T('\\\\');\n            m_szDumpFilePath[3] = _T('\\0');\n        }\n    }\nprivate:\n    wchar_t m_szDumpFilePath[MAX_PATH];\n    wchar_t m_szModuleFileName[MAX_PATH];\n\n\tCString m_dumpFile;\n};\n\nnamespace CRASHREPORT\n{\n    static LONG WINAPI __UnhandledExceptionFilter(PEXCEPTION_POINTERS pEP)\n    {\n        ::SetErrorMode(0); //使用默认的\n        CCrashReport cr;\n        cr.CreateMiniDump(pEP);\n\t\tcr.ShowCrashInfo();\n        return EXCEPTION_CONTINUE_SEARCH;\n    }\n\n    void StartCrashReport()\n    {\n        ::SetUnhandledExceptionFilter(__UnhandledExceptionFilter);\n    }\n}\n"
  },
  {
    "path": "TrafficMonitor/crashtool.h",
    "content": "#pragma once\n\nnamespace CRASHREPORT\n{\n    /**@brief\n        dmupģʽ\n        󣬵ʱһdumpļļC:\\Users\\<û>\\AppData\\Local\\TempĿ¼\n        ļʽʱ_.exe.dmp\n        \n        20150116174802_CrashShare.exe.dmp\n    */\n    void StartCrashReport();\n}"
  },
  {
    "path": "TrafficMonitor/print_compile_time.bat",
    "content": "REM ǰںʱ䣬浽compile_time.txtļ\ndel /F /Q compile_time.txt\necho %date:~0,10% >> compile_time.txt\necho %time:~0,8% >> compile_time.txt"
  },
  {
    "path": "TrafficMonitor/resource.h",
    "content": "//{{NO_DEPENDENCIES}}\n// Microsoft Visual C++ ɵİļ\n//  TrafficMonitor.rc ʹ\n//\n#define IDD_TRAFFICMONITOR_DIALOG       102\n#define IDR_MAINFRAME                   128\n#define IDR_MENU1                       130\n#define IDD_NETWORK_INFO_DIALOG         131\n#define IDD_ABOUTBOX                    133\n#define IDD_TASK_BAR_DIALOG             135\n#define IDR_TASK_BAR_MENU               137\n#define IDD_SKIN_DIALOG                 138\n#define IDB_BITMAP1                     140\n#define IDD_HISTORY_TRAFFIC_DIALOG      141\n#define IDD_DONATE_DIALOG               143\n#define IDB_BITMAP2                     145\n#define IDB_DONATE_BITMAP               145\n#define IDI_NOFITY_ICON                 146\n#define IDD_OPTIONS_DIALOG              147\n#define IDD_MAIN_WND_SETTINGS_DIALOG    149\n#define IDD_TASKBAR_SETTINGS_DIALOG     150\n#define IDR_INFO_MENU                   153\n#define IDR_MENU2                       154\n#define IDR_HISTORY_TRAFFIC_MENU        154\n#define IDD_GENERAL_SETTINGS_DIALOG     155\n#define IDB_ABOUT_BACKGROUND_HD         157\n#define IDI_NOFITY_ICON2                158\n#define IDI_NOFITY_ICON3                159\n#define IDD_ICON_SELECT_DIALOG          160\n#define IDB_NOTIFY_ICON_PREVIEW         162\n#define IDD_MAIN_COLOR_DIALOG           163\n#define IDD_TASKBAR_COLOR_DIALOG        165\n#define IDS_CHECK_UPDATE_FAILD          167\n#define IDS_CHECK_UPDATE_ERROR          168\n#define IDD_HISTORY_TRAFFIC_LIST_DIALOG 168\n#define IDS_UPDATE_AVLIABLE             169\n#define IDS_UPDATE_AVLIABLE2            170\n#define IDD_HISTORY_TRAFFIC_CALENDAR_DIALOG 170\n#define IDS_ALREADY_UPDATED             171\n#define IDS_AUTORUN_FAILED_NO_KEY       172\n#define IDS_AUTORUN_FAILED_NO_ACCESS    173\n#define IDS_AUTORUN_DELETE_FAILED       174\n#define IDS_AN_INSTANCE_RUNNING         175\n#define IDS_TRAFFIC_USED_TODAY          176\n#define IDS_MEMORY_USAGE                177\n#define IDS_CPU_USAGE                   178\n#define IDS_SEND_EMAIL_TO_ATHOUR        179\n#define IDS_GOTO_GITHUB                 180\n#define IDS_DONATE_ATHOUR               181\n#define IDS_UPLOAD                      182\n#define IDS_DOWNLOAD                    183\n#define IDS_MEMORY                      184\n#define IDS_UPLOAD_DISP                 185\n#define IDS_DOWNLOAD_DISP               186\n#define IDS_MEMORY_DISP                 187\n#define IDS_CONNOT_SAVE_CONFIG_WARNING  188\n#define IDS_TRAFFICMONITOR              189\n#define IDS_INSUFFICIENT_BUFFER         190\n#define IDS_CONNECTION_NUM_CHANGED      191\n#define IDS_CONNECTION_NOT_MATCH        192\n#define IDS_CONNOT_INSERT_TO_TASKBAR    193\n#define IDS_MEMORY_UDAGE_EXCEED         194\n#define IDS_NOTIFY                      195\n#define IDS_TODAY_TRAFFIC_EXCEED        196\n#define IDS_TITLE_ABOUT                 197\n#define IDS_TITLE_DONATE                198\n#define IDS_TITLE_HISTORY_TRAFFIC       199\n#define IDS_TITLE_CONNECTION_DETIAL     200\n#define IDS_TITLE_CHANGE_SKIN           201\n#define IDS_TITLE_OPTION                202\n#define IDS_TITLE_CHANGE_ICON           203\n#define IDS_DATE                        204\n#define IDS_TRAFFIC_USED                205\n#define IDS_FIGURE                      206\n#define IDS_DEFAULT_ICON                207\n#define IDS_ICON                        208\n#define IDS_INTERFACE_NAME              209\n#define IDS_INTERFACE_DESCRIPTION       210\n#define IDS_CONNECTION_TYPE             211\n#define IDS_IF_TYPE_OTHER               212\n#define IDS_IF_TYPE_ETHERNET_CSMACD     213\n#define IDS_IF_TYPE_ISO88025_TOKENRING  214\n#define IDS_IF_TYPE_FDDI                215\n#define IDS_IF_TYPE_PPP                 216\n#define IDS_IF_TYPE_SOFTWARE_LOOPBACK   217\n#define IDS_IF_TYPE_ATM                 218\n#define IDS_IF_TYPE_IEEE80211           219\n#define IDS_IF_TYPE_TUNNEL              220\n#define IDS_IF_TYPE_IEEE1394            221\n#define IDS_IF_TYPE_IEEE80216_WMAN      222\n#define IDS_IF_TYPE_WWANPP              223\n#define IDS_IF_TYPE_WWANPP2             224\n#define IDS_UNKNOW_CONNECTION           225\n#define IDS_SPEED                       226\n#define IDS_ADAPTER_PHYSICAL_ADDRESS    227\n#define IDS_IP_ADDRESS                  228\n#define IDS_SUBNET_MASK                 229\n#define IDS_DEFAULT_GATEWAY             230\n#define IDS_OPERATIONAL_STATUS          231\n#define IDS_IF_OPER_STATUS_NON_OPERATIONAL 232\n#define IDS_IF_OPER_STATUS_UNREACHABLE  233\n#define IDS_IF_OPER_STATUS_DISCONNECTED 234\n#define IDS_IF_OPER_STATUS_CONNECTING   235\n#define IDS_IF_OPER_STATUS_CONNECTED    236\n#define IDS_IF_OPER_STATUS_OPERATIONAL  237\n#define IDS_UNKNOW_STATUS               238\n#define IDS_BYTES_RECEIVED              239\n#define IDS_BYTES_SENT                  240\n#define IDS_BYTES_RECEIVED_SINCE_START  241\n#define IDS_BYTES_SENT_SINCE_START      242\n#define IDS_PROGRAM_ELAPSED_TIME        243\n#define IDS_HOUR_MINUTE_SECOND          244\n#define IDS_INTERNET_IP_ADDRESS         245\n#define IDS_GET_FAILED                  246\n#define IDS_ITEM                        247\n#define IDS_VALUE                       248\n#define IDS_COPY_TO_CLIPBOARD_FAILED    249\n#define IDS_SKIN_AUTHOUR                250\n#define IDS_OVERWRITE_FONT_TEXT_WARNING 251\n#define IDS_OVERWRITE_FONT_WARNING      252\n#define IDS_OVERWRITE_TEXT_WARNING      253\n#define IDS_SPEED_SHORT_MODE_TIP        254\n#define IDS_AUTO                        255\n#define IDS_FIXED_AS                    256\n#define IDS_OPEN_CONNECTION_DETIAL      257\n#define IDS_OPEN_HISTORICAL_TRAFFIC     258\n#define IDS_SHOW_HIDE_MORE_INFO         259\n#define IDS_SHOW_HIDE_CPU_MEMORY        260\n#define IDS_OPEN_OPTION_SETTINGS        261\n#define IDS_OPEN_TASK_MANAGER           262\n#define IDS_CHANGE_SKIN                 263\n#define IDS_NONE                        264\n#define IDS_FONT_SIZE_WARNING           265\n#define IDS_SAME_TEXT_BACK_COLOR_WARNING 266\n#define IDS_SAME_BACK_TEXT_COLOR_WARNING 267\n#define IDS_FOLLOWING_SYSTEM            268\n#define IDS_LANGUAGE_CHANGE_INFO        269\n#define IDS_MAIN_WINDOW_SETTINGS        270\n#define IDS_TASKBAR_WINDOW_SETTINGS     271\n#define IDS_GENERAL_SETTINGS            272\n#define IDS_MICROSOFT_YAHEI             273\n#define IDS_ACQUIRING                   274\n#define IDS_DEFAULT_FONT                275\n#define IDS_LIST_VIEW                   276\n#define IDS_CALENDAR_VIEW               277\n#define IDS_MONDAY                      278\n#define IDS_TUESDAY                     279\n#define IDS_WEDNESDAY                   280\n#define IDS_THURSDAY                    281\n#define IDS_FRIDAY                      282\n#define IDS_SATURDAY                    283\n#define IDS_SUNDAY                      284\n#define IDS_CURRENT_MONTH_TOTAL_TRAFFIC 285\n#define IDS_TRAFFIC_USED1               286\n#define IDS_LANGUAGE_CODE               287\n#define IDS_CONNOT_INSERT_TO_TASKBAR_ERROR_LOG 288\n#define IDS_NO_CONNECTION               289\n#define IDS_SEND_EMAIL_TO_TRANSLATOR    290\n#define IDS_CONTACT_TRANSLATOR          290\n#define IDR_TEXT2                       290\n#define IDR_TEXT                        290\n#define IDS_THANKS_DONORS               291\n#define IDS_GET_URL_ERROR_LOG_INFO      292\n#define IDS_SHOW_ALL_INFO_TIP           293\n#define IDD_DIALOG1                     293\n#define IDD_MESSAGE_DIALOG              293\n#define IDR_ACKNOWLEDGEMENT_TEXT        294\n#define IDS_CFG_DIR_CHANGED_INFO        294\n#define IDS_DOUBLE_CLICK_TO_ACQUIRE     295\n#define IDS_ERROR1                      296\n#define IDS_ERROR_MESSAGE               297\n#define IDS_CRASH_INFO                  298\n#define IDS_TITLE_ACKNOWLEDGEMENT       299\n#define IDI_NOFITY_ICON4                299\n#define IDS_SAVE_DEFAULT_STYLE_INQUIRY  300\n#define IDS_SPECIFIC_APP                301\n#define IDB_NOTIFY_ICON_PREVIEW_LIGHT   302\n#define IDS_EXE_FILTER                  302\n#define IDS_PRESET                      303\n#define IDS_LIGHT_MODE                  304\n#define IDB_BITMAP3                     305\n#define IDB_DONATE_WECHAT               305\n#define IDS_AUTO_ADAPT_TIP_INFO         305\n#define IDD_ATUO_ADAPT_SETTING_DIALOG   306\n#define IDS_WITHOUT_TEMPERATURE         306\n#define IDS_MOUSE_PENETRATE_TIP_INFO    307\n#define IDS_HISTORY_TRAFFIC_LOST_ERROR_LOG 308\n#define IDS_LEGEND                      309\n#define IDR_LICENSE                     310\n#define IDS_LICENSE_EXPLAIN             310\n#define IDS_LICENSE                     311\n#define IDI_EXIT                        311\n#define IDI_HELP                        312\n#define IDS_DAY_VIEW                    312\n#define IDI_INFO                        313\n#define IDS_MONTH_VIEW                  313\n#define IDI_LOCK                        314\n#define IDS_QUARTER_VIEW                314\n#define IDI_SETTINGS                    315\n#define IDS_YEAR_VIEW                   315\n#define IDI_STATISTICS                  316\n#define IDS_LINEAR_SCALE                316\n#define IDI_CLOSE                       317\n#define IDS_LOG_SCALE                   317\n#define IDI_PIN                         318\n#define IDS_CPU_TEMPERATURE             318\n#define IDI_SKIN                        319\n#define IDS_GPU_TEMPERATURE             319\n#define IDI_CONNECTION                  320\n#define IDS_HDD_TEMPERATURE             320\n#define IDI_MOUSE                       321\n#define IDS_MAINBOARD_TEMPERATURE       321\n#define IDI_MAIN_WINDOW                 322\n#define IDS_GPU_DISP                    322\n#define IDI_TASKBAR_WINDOW              323\n#define IDS_HDD_DISP                    323\n#define IDI_NOTIFY                      324\n#define IDS_MAINBOARD_DISP              324\n#define IDI_MORE                        325\n#define IDS_CPU_TEMPERATURE_EXCEED      325\n#define IDI_ITEM                        326\n#define IDS_GPU_TEMPERATURE_EXCEED      326\n#define IDI_FUNCTION                    327\n#define IDS_HDD_TEMPERATURE_EXCEED      327\n#define IDI_NOTIFY_ICON5                328\n#define IDS_MBD_TEMPERATURE_EXCEED      328\n#define IDD_DISPLAY_TEXT_SETTINGS_DIALOG 329\n#define IDS_MUSICPLAYER2_DESCRIPTION    329\n#define IDR_COMPILE_TIME                329\n#define IDS_SIMPLENOTEPAD_DESCRIPTION   330\n#define IDD_SELECT_ORDER_DIALOG         330\n#define IDS_COLOR                       331\n#define IDS_COLOR_LABEL                 332\n#define IDD_PLUGIN_MANAGER_DIALOG       332\n#define IDS_COLOR_VALUE                 333\n#define IDS_GPU_USAGE                   334\n#define IDI_PLUGINS                     334\n#define IDS_IF_OPER_STATUS_UP           335\n#define IDR_MENU3                       335\n#define IDR_PLUGIN_MANAGER_MENU         335\n#define IDS_IF_OPER_STATUS_DOWN         336\n#define IDI_TASK_MANAGER                336\n#define IDS_IF_OPER_STATUS_DORMANT      337\n#define IDD_APP_ALREAD_RUNING_DIALOG    337\n#define IDS_GOTO_GITEE                  338\n#define IDS_USAGE_PERCENTAGE            339\n#define IDD_SELECT_CONNECTIONS_DIALOG   339\n#define IDS_MEMORY_USED                 340\n#define IDS_MEMORY_AVAILABLE            341\n#define IDS_DOTNET_NOT_INSTALLED_TIP    342\n#define IDS_VERSION_UPDATE              343\n#define IDS_AVREAGE_TEMPERATURE         344\n#define IDS_HARDWARE_MONITOR_WARNING    345\n#define IDS_HARDWARE_MONITOR_WARNING2   346\n#define IDS_HDD_USAGE                   347\n#define IDS_FILE_NAME                   348\n#define IDS_STATUS                      349\n#define IDS_PLUGIN_LOAD_SUCCEED         350\n#define IDS_PLUGIN_MODULE_LOAD_FAILED   351\n#define IDS_PLUGIN_FUNCTION_GET_FAILED  352\n#define IDS_PLUGIN_INFO                 353\n#define IDS_NAME                        354\n#define IDS_DESCRIPTION                 355\n#define IDS_FILE_PATH                   356\n#define IDS_ITEM_NUM                    357\n#define IDS_ITEM_NAMES                  358\n#define IDS_AUTHOR                      359\n#define IDS_COPYRIGHT                   360\n#define IDS_PLUGIN_NO_OPTIONS_INFO      361\n#define IDS_PLUGIN_NAME                 362\n#define IDS_DISABLED                    363\n#define IDS_RESTART_TO_APPLY_CHANGE_INFO 364\n#define IDS_VERSION                     365\n#define IDS_DISP_ITEM_ID                366\n#define IDS_PLUGIN_API_VERSION          367\n#define IDS_WEEK_VIEW                   368\n#define IDS_WEEK_NUM                    369\n#define IDS_URL                         370\n#define IDS_PLUGIN_VERSION_NOT_SUPPORT  371\n#define IDS_MODIFY_PRESET               372\n#define IDS_SELECT_AT_LEASE_ONE_WARNING 373\n#define IDS_AUTO_SAVE_TO_PRESET_TIP     374\n#define IDS_TOTAL_NET_SPEED             375\n#define IDS_SHOW_RESOURCE_USAGE_GRAPH_TIP 376\n#define IDS_SHOW_NET_SPEED_GRAPH_TIP    377\n#define IDS_REFRESH_CONNECTION_LIST     378\n#define IDS_HARDWARE_MONITOR_INIT_FAILED 379\n#define IDS_HARDWARE_INFO_ACQUIRE_FAILED_ERROR 380\n#define IDS_AUTO_RUN_METHOD_REGESTRY    381\n#define IDS_AUTO_RUN_METHOD_TASK_SCHEDULE 382\n#define IDS_PATH                        383\n#define IDS_SET_AUTO_RUN_FAILED_WARNING 384\n#define IDC_STATIC_INFO                 1001\n#define IDC_STATIC1                     1002\n#define IDC_STATIC_DOWN                 1002\n#define IDC_STATIC_UP                   1003\n#define IDC_STATIC2                     1003\n#define IDC_STATIC_DOWN1                1003\n#define IDC_STATIC_MEMORY               1004\n#define IDC_STATIC3                     1004\n#define IDC_STATIC_CPU                  1005\n#define IDC_STATIC4                     1005\n#define IDC_STATIC_UP1                  1006\n#define IDC_STATIC_VERSION              1007\n#define IDC_COMBO1                      1008\n#define IDC_STATIC_SKIN_S               1009\n#define IDC_STATIC_SKIN_L               1010\n#define IDC_STATIC_TEXT2                1012\n#define IDC_SKIN_INFO                   1013\n#define IDC_STATIC_ABOUT                1015\n#define IDC_STATIC_MAIL                 1016\n#define IDC_LIST1                       1017\n#define IDC_INFO_LIST                   1017\n#define IDC_INFO_LIST1                  1017\n#define IDC_HISTORY_INFO_LIST           1017\n#define IDC_STATIC_LICENSE              1017\n#define IDC_DONATE_PIC                  1018\n#define IDC_FONT_NAME_EDIT              1019\n#define IDC_FONT_SIZE_EDIT              1020\n#define IDC_STATIC_CHECK_UPDATE         1021\n#define IDC_SET_FONT_BUTTON             1021\n#define IDC_STATIC_DONATE               1022\n#define IDC_TEXT_COLOR_STATIC           1022\n#define IDC_UPLOAD_EDIT                 1023\n#define IDC_DOWNLOAD_EDIT               1024\n#define IDC_CPU_EDIT                    1025\n#define IDC_MEMORY_EDIT                 1026\n#define IDC_FONT_NAME_EDIT1             1027\n#define IDC_FONT_SIZE_EDIT1             1028\n#define IDC_SET_FONT_BUTTON1            1029\n#define IDC_TEXT_COLOR_STATIC1          1030\n#define IDC_UPLOAD_EDIT1                1031\n#define IDC_DOWNLOAD_EDIT1              1032\n#define IDC_CPU_EDIT1                   1033\n#define IDC_MEMORY_EDIT1                1034\n#define IDC_TEXT_COLOR_STATIC2          1035\n#define IDC_TAB1                        1035\n#define IDC_SET_DEFAULT_BUTTON          1036\n#define IDC_TEXT_COLOR_STATIC3          1036\n#define IDC_SET_COLOR_BUTTON2           1037\n#define IDC_TRANSPARENT_COLOR_STATIC    1037\n#define IDC_SET_COLOR_BUTTON1           1038\n#define IDC_SET_COLOR_BUTTON3           1038\n#define IDC_SWITCH_UP_DOWN_CHECK        1039\n#define IDC_SWITCH_UP_DOWN_CHECK1       1040\n#define IDC_SET_DEFAULT_BUTTON1         1041\n#define IDC_FULLSCREEN_HIDE_CHECK       1042\n#define IDC_SWITCH_UP_DOWN_CHECK2       1042\n#define IDC_SPEED_SHORT_MODE_CHECK      1042\n#define IDC_STATIC_GITHUB               1043\n#define IDC_SPEED_SHORT_MODE_CHECK2     1043\n#define IDC_SET_LIGHT_MODE_BUTTON       1043\n#define IDC_CHECK_UPDATE_CHECK          1044\n#define IDC_STATIC_GITEE                1044\n#define IDC_CHECK_NOW_BUTTON            1045\n#define IDC_AUTO_RUN_CHECK              1046\n#define IDC_TASKBAR_WND_ON_LEFT_CHECK   1047\n#define IDC_HIDE_UNIT_CHECK             1049\n#define IDC_UNIT_COMBO                  1050\n#define IDC_ICON_PREVIEW                1051\n#define IDC_HIDE_PERCENTAGE_CHECK       1051\n#define IDC_SKIN_COURSE_STATIC          1052\n#define IDC_SKIN_COURSE_STATIC2         1053\n#define IDC_SKIN_DOWNLOAD_STATIC        1053\n#define IDC_PREVIEW_GROUP_STATIC        1054\n#define IDC_NOTIFY_STATIC               1056\n#define IDC_ALLOW_SKIN_FONT_CHECK       1057\n#define IDC_CHECK2                      1058\n#define IDC_ALLOW_SKIN_DISP_STR_CHECK   1058\n#define IDC_VALUE_RIGHT_ALIGN_CHECK     1058\n#define IDC_ALWAYS_ON_TOP_CHECK         1058\n#define IDC_UP_STATIC                   1059\n#define IDC_DOWN_STATIC                 1060\n#define IDC_UP_VALUE_STATIC             1060\n#define IDC_CPU_STATIC                  1061\n#define IDC_DWON_LABLE_STATIC           1061\n#define IDC_MEMORY_STATIC               1062\n#define IDC_DWON_VALUE_STATIC           1062\n#define IDC_SPECIFY_EACH_ITEM_COLOR_CHECK 1063\n#define IDC_CPU_LABLE_STATIC            1063\n#define IDC_CPU_TEMP_STATIC             1063\n#define IDC_CPU_VALUE_STATIC            1064\n#define IDC_GPU_TEMP_STATIC             1064\n#define IDC_MEMORY_LABLE_STATIC         1065\n#define IDC_HDD_TEMP_STATIC             1065\n#define IDC_UP_STATIC8                  1066\n#define IDC_MEMORY_VALUE_STATIC         1066\n#define IDC_MEMORY_STATIC2              1066\n#define IDC_MBD_TEMP_STATIC             1066\n#define IDC_UP_LABLE_STATIC             1067\n#define IDC_DOUBLE_CLICK_COMBO          1068\n#define IDC_TODAY_TRAFFIC_TIP_CHECK     1069\n#define IDC_TODAY_TRAFFIC_TIP_EDIT      1070\n#define IDC_MEMORY_USAGE_TIP_CHECK      1071\n#define IDC_MEMORY_USAGE_TIP_EDIT       1072\n#define IDC_TODAY_TRAFFIC_TIP_COMBO     1073\n#define IDC_STATIC_COPYRIGHT            1074\n#define IDC_LANGUAGE_COMBO              1075\n#define IDC_CPU_TEMP_LABLE_STATIC       1076\n#define IDC_CPU_TEMP_TIP_EDIT           1076\n#define IDC_YEAR_COMBO                  1077\n#define IDC_CPU_TEMP_VALUE_STATIC       1077\n#define IDC_GPU_TEMP_TIP_CHECK          1077\n#define IDC_MONTH_COMBO                 1078\n#define IDC_GPU_TEMP_LABLE_STATIC       1078\n#define IDC_GPU_TEMP_TIP_EDIT           1078\n#define IDC_PREVIOUS_BUTTON             1079\n#define IDC_GPU_TEMP_VALUE_STATIC       1079\n#define IDC_NEXT_BUTTON                 1080\n#define IDC_OPEN_CONFIG_PATH_BUTTON     1080\n#define IDC_HDD_TEMP_LABLE_STATIC       1080\n#define IDC_DIGIT_NUMBER_COMBO          1081\n#define IDC_HDD_TEMP_VALUE_STATIC       1081\n#define IDC_HDD_TIP_EDIT                1081\n#define IDC_HORIZONTAL_ARRANGE_CHECK    1082\n#define IDC_MAIN_BOARD_TEMP_LABLE_STATIC 1082\n#define IDC_MBD_TEMP_TIP_CHECK          1082\n#define IDC_PREVIEW_BUTTON              1083\n#define IDC_MAIN_BOARD_TEMP_VALUE_STATIC 1083\n#define IDC_MBD_TEMP_TIP_EDIT           1083\n#define IDC_INDEX_STATIC                1084\n#define IDC_STATIC_ACKNOWLEDGEMENT      1085\n#define IDC_TRADITIONAL_CHINESE_TRANSLATOR_STATIC 1086\n#define IDC_TRANSLATOR_STATIC           1087\n#define IDC_SEPARATE_VALUE_UNIT_CHECK   1089\n#define IDC_SHOW_STATUS_BAR_CHECK       1090\n#define IDC_UNIT_BYTE_RADIO             1091\n#define IDC_UNIT_BIT_RADIO              1092\n#define IDC_SHOW_ALL_CONNECTION_CHECK   1092\n#define IDC_SAVE_TO_APPDATA_RADIO       1093\n#define IDC_SAVE_TO_PROGRAM_DIR_RADIO   1094\n#define IDC_HELP_EDIT                   1095\n#define IDC_INFO_STATIC                 1096\n#define IDC_SHOW_TOOL_TIP_CHK           1097\n#define IDC_DEFAULT_STYLE_BUTTON        1098\n#define IDC_BROWSE_BUTTON               1100\n#define IDC_EXE_PATH_EDIT               1101\n#define IDC_EXE_PATH_STATIC             1102\n#define IDC_TEXT_STATIC                 1103\n#define IDC_CM_GRAPH_BAR_RADIO          1104\n#define IDC_CM_GRAPH_PLOT_RADIO         1105\n#define IDC_USE_CPU_TIME_RADIO          1106\n#define IDC_USE_PDH_RADIO               1107\n#define IDC_BACKGROUND_TRANSPARENT_CHECK 1108\n#define IDC_AUTO_ADAPT_LIGHT_THEME_CHECK 1109\n#define IDC_AUTO_ADAPT_SETTINGS_BUTTON  1110\n#define IDC_DARK_MODE_DEFAULT_STYLE_COMBO 1111\n#define IDC_LIGHT_MODE_DEFAULT_STYLE_COMBO 1112\n#define IDC_MENU_BUTTON                 1112\n#define IDC_AUTO_ADAPT_CHECK            1113\n#define IDC_MONITOR_SPAN_EDIT           1114\n#define IDC_VIEW_TYPE_COMBO             1115\n#define IDC_VIEW_SCALE_COMBO            1116\n#define IDC_HDD_TEMP_TIP_CHECK          1117\n#define IDC_AUTO_SET_BACK_COLOR_CHECK   1117\n#define IDC_RESTORE_DEFAULT_BUTTON      1118\n#define IDC_DISPLAY_TEXT_SETTING_BUTTON 1119\n#define IDC_CPU_TEMP_TIP_CHECK          1120\n#define IDC_OPENHARDWAREMONITOR_LINK    1122\n#define IDC_TINYXML2_LINK               1123\n#define IDC_MUSICPLAYER2_LINK           1124\n#define IDC_SIMPLENOTEPAD_LINK          1125\n#define IDC_GITHUB_RADIO                1126\n#define IDC_GITEE_RADIO                 1127\n#define IDC_MEMORY_DISPLAY_COMBO        1127\n#define IDC_RESTORE_DEFAULT_TIME_SPAN_BUTTON 1128\n#define IDC_SELECT_HARD_DISK_COMBO      1129\n#define IDC_CPU_CHECK                   1130\n#define IDC_GPU_CHECK                   1131\n#define IDC_SHOW_DASHED_BOX             1131\n#define IDC_HDD_CHECK                   1132\n#define IDC_MBD_CHECK                   1133\n#define IDC_MOUSE_PENETRATE_CHECK       1133\n#define IDC_SELECT_CPU_COMBO            1134\n#define IDC_LOCK_WINDOW_POS_CHECK       1134\n#define IDC_ALOW_OUT_OF_BORDER_CHECK    1135\n#define IDC_MOVE_UP_BUTTON              1136\n#define IDC_MOVE_DOWN_BUTTON            1137\n#define IDC_SET_ORDER_BUTTON            1137\n#define IDC_OPTINS_BUTTON               1139\n#define IDC_PLUGIN_MANAGE_BUTTON        1140\n#define IDC_PLUGIN_INFO_BUTTON          1141\n#define IDC_SHOW_NOTIFY_ICON_CHECK      1142\n#define IDC_HDD_STATIC                  1143\n#define IDC_HARDWARE_MONITOR_STATIC     1144\n#define IDC_SELECT_HDD_STATIC           1145\n#define IDC_SELECT_CPU_STATIC           1146\n#define IDC_APPLY_BUTTON                1147\n#define IDC_TASKBAR_WND_SNAP_CHECK      1148\n#define IDC_ITEM_SPACE_EDIT             1149\n#define IDC_EXIT_INST_BUTTON            1150\n#define IDC_OPEN_SETTINGS_BUTTON        1151\n#define IDC_SHOW_HIDE_MAIN_WINDOW_BUTTON 1152\n#define IDC_SHOW_HIDE_TASKBAR_WINDOW_BUTTON 1153\n#define IDC_PLUGIN_DEV_GUID_STATIC      1153\n#define IDC_PLUGIN_DOWNLOAD_STATIC      1154\n#define IDC_SELECT_CONNECTIONS_BUTTON   1155\n#define IDC_AUTO_SAVE_TO_PRESET_CHECK   1156\n#define IDC_SHOW_NET_SPEED_FIGURE_CHECK 1157\n#define IDC_NET_SPEED_FIGURE_MAX_VALUE_EDIT 1158\n#define IDC_NET_SPEED_FIGURE_MAX_VALUE_UNIT_COMBO 1159\n#define IDC_OPEN_PLUGIN_DIR_STATIC      1160\n#define IDC_OPEN_SKIN_DIR_STATIC        1161\n#define IDC_RESET_AUTO_RUN_BUTTON       1162\n#define IDS_CPU_FREQ\t\t\t\t\t1163\n\n#define ID_32771                        32771\n#define ID_NETWORK_INFO                 32772\n#define ID_32773                        32773\n#define ID_ALWAYS_ON_TOP                32774\n#define ID_32775                        32775\n#define ID_32776                        32776\n#define ID_32777                        32777\n#define ID_32778                        32778\n#define ID_32779                        32779\n#define ID_TRANS_100                    32780\n#define ID_TRANSPARENCY_100             32781\n#define ID_TRANSPARENCY_80              32782\n#define ID_TRANSPARENCY_60              32783\n#define ID_TRANSPARENCY_40              32784\n#define ID_32785                        32785\n#define ID_32786                        32786\n#define ID_32789                        32789\n#define ID_LOCK_WINDOW_POS              32790\n#define ID_32791                        32791\n#define ID_SHOW_NOTIFY_ICON             32792\n#define ID_32793                        32793\n#define ID_SHOW_CPU_MEMORY              32794\n#define ID_32795                        32795\n#define ID_MOUSE_PENETRATE              32796\n#define ID_32797                        32797\n#define ID_32798                        32798\n#define ID_TEXT_COLOR                   32799\n#define ID_32800                        32800\n#define ID_SHOW_TASK_BAR_WND            32801\n#define ID_32802                        32802\n#define ID_32803                        32803\n#define ID_32804                        32804\n#define ID_32805                        32805\n#define ID_32806                        32806\n#define ID_SET_BACK_COLOR               32807\n#define ID_32808                        32808\n#define ID_SET_TEXT_COLOR               32809\n#define ID_32810                        32810\n#define ID_SHOW_CPU_MEMORY2             32811\n#define ID_32812                        32812\n#define ID_MINMIZE                      32813\n#define ID_32814                        32814\n#define ID_AUTO_RUN_WHEN_START          32815\n#define ID_HIDE_MAIN_WND                32816\n#define ID_32817                        32817\n#define ID_32818                        32818\n#define ID_CHANGE_SKIN                  32819\n#define ID_32820                        32820\n#define ID_SET_FONT                     32821\n#define ID_32822                        32822\n#define ID_SET_                         32823\n#define ID_SET_FONT2                    32824\n#define ID_32825                        32825\n#define ID_TRAFFIC_HISTORY              32826\n#define ID_32827                        32827\n#define ID_32828                        32828\n#define ID_OPTIONS                      32829\n#define ID_32830                        32830\n#define ID_OPTIONS2                     32831\n#define ID_32832                        32832\n#define ID_COPY_TEXT                    32833\n#define ID_32834                        32834\n#define ID_32835                        32835\n#define ID_USE_LINEAR_SCALE             32836\n#define ID_USE_LOG_SCALE                32837\n#define ID_32838                        32838\n#define ID_CHANGE_NOTIFY_ICON           32839\n#define ID_32840                        32840\n#define ID_ALOW_OUT_OF_BORDER           32841\n#define ID_32842                        32842\n#define ID_32843                        32843\n#define ID_32844                        32844\n#define ID_32845                        32845\n#define ID_CHECK_UPDATE                 32846\n#define ID_SHOW_MAIN_WND                32847\n#define ID_DEFAULT_STYLE1               32848\n#define ID_DEFAULT_STYLE_MAX            32857\n#define ID_MODIFY_DEFAULT_STYLE1        32858\n#define ID_MODIFY_DEFAULT_STYLE_MAX     32867\n#define ID_LIGHT_MODE_STYLE             32868\n#define ID_SHOW_NET_SPEED               32869\n#define ID_FIRST_DAY_OF_WEEK_SUNDAY     32870\n#define ID_FIRST_DAY_OF_WEEK_MONDAY     32871\n#define ID_CALENDAR_JUMP_TO_TODAY       32872\n#define ID_FREQUENTY_ASKED_QUESTIONS    32873\n#define ID_CMD_TEST                     32874\n#define ID_SHOW_UP_SPEED                32875\n#define ID_SHOW_DOWN_SPEED              32876\n#define ID_SHOW_CPU_USAGE               32877\n#define ID_SHOW_MEMORY_USAGE            32878\n#define ID_SHOW_CPU_TEMPERATURE         32879\n#define ID_SHOW_GPU_TEMPERATURE         32880\n#define ID_SHOW_HDD_TEMPERATURE         32881\n#define ID_SHOW_MAIN_BOARD_TEMPERATURE  32882\n#define ID_SHOW_GPU                     32883\n#define ID_SHOW_HDD                     32884\n#define ID_UPDATE_LOG                   32885\n#define ID_PLUGIN_MANAGE                32886\n#define ID_SHOW_PLUGIN_ITEM_START       32887\n#define ID_SHOW_PLUGIN_ITEM_MAX         33140\n#define ID_SELETE_CONNECTION            33141\n#define ID_SELECT_ALL_CONNECTION        33142\n#define ID_SELETE_CONNECTION_MAX        33398\n#define ID_PLUGIN_DETAIL                33399\n#define ID_PLUGIN_OPTIONS               33400\n#define ID_PLUGIN_DISABLE               33401\n#define ID_SHOW_TOTAL_SPEED             33402\n#define ID_OPEN_TASK_MANAGER            33403\n#define ID_REFRESH_CONNECTION_LIST      33404\n#define ID_DISPLAY_SETTINGS             33405\n\n// Next default values for new objects\n// \n#ifdef APSTUDIO_INVOKED\n#ifndef APSTUDIO_READONLY_SYMBOLS\n#define _APS_NEXT_RESOURCE_VALUE        341\n#define _APS_NEXT_COMMAND_VALUE         33406\n#define _APS_NEXT_CONTROL_VALUE         1166\n#define _APS_NEXT_SYMED_VALUE           101\n#endif\n#endif\n"
  },
  {
    "path": "TrafficMonitor/skins/0默认皮肤/skin.ini",
    "content": "[skin]\ntext_color=16384\nskin_author=zy"
  },
  {
    "path": "TrafficMonitor/skins/xml_test/skin.xml",
    "content": "﻿<root>\n\t<skin>\n\t\t<text_color>16768959,10022341,16777215,16777215,16777215,16770992,16770992,16770992,16770992</text_color>\n\t\t<specify_each_item_color>1</specify_each_item_color>\n\t\t<skin_author>zy</skin_author>\n\t\t<font name=\"Microsoft YaHei\" size=\"9\" style=\"0\"/>\n\t\t<display_text>\n\t\t\t<up>上传: </up>\n\t\t\t<down>下载: </down>\n\t\t\t<cpu>CPU: </cpu>\n\t\t\t<memory>内存: </memory>\n\t\t\t<gpu>显卡: </gpu>\n\t\t\t<cpu_temperature>CPU温度: </cpu_temperature>\n\t\t\t<gpu_temperature>显卡温度: </gpu_temperature>\n\t\t\t<total_speed>↑↓: </total_speed>\n\t\t</display_text>\n\t</skin>\n\t<plugin_map>\n\t\t<time>ra1YX2g1</time>\n\t\t<cosutom_item>b4zc373y</cosutom_item>\n\t\t<battery>b5R30ITQ</battery>\n\t</plugin_map>\n\t<layout text_height=\"20\" no_label=\"0\">\n\t\t<layout_l width=\"232\" height=\"107\">\n\t\t\t<cpu x=\"14\" y=\"3\" width=\"106\" align=\"0\" show=\"1\"/>\n\t\t\t<memory x=\"88\" y=\"3\" width=\"106\" align=\"0\" show=\"1\"/>\n\t\t\t<battery x=\"162\" y=\"3\" width=\"106\" align=\"0\" show=\"1\"/>\n\t\t\t<up x=\"14\" y=\"27\" width=\"106\" align=\"0\" show=\"1\"/>\n\t\t\t<down x=\"122\" y=\"27\" width=\"106\" align=\"0\" show=\"1\"/>\n\t\t\t<total_speed x=\"32\" y=\"57\" width=\"94\" align=\"0\" show=\"1\"/>\n\t\t\t<cpu_temperature x=\"32\" y=\"83\" width=\"94\" align=\"0\" show=\"1\"/>\n\t\t\t<time x=\"142\" y=\"57\" width=\"94\" align=\"0\" show=\"1\"/>\n\t\t\t<cosutom_item x=\"142\" y=\"83\" width=\"94\" align=\"0\" show=\"1\"/>\n\t\t</layout_l>\n\t\t<layout_s width=\"232\" height=\"50\">\n\t\t\t<cpu x=\"14\" y=\"3\" width=\"106\" align=\"0\" show=\"1\"/>\n\t\t\t<memory x=\"88\" y=\"3\" width=\"106\" align=\"0\" show=\"1\"/>\n\t\t\t<battery x=\"162\" y=\"3\" width=\"106\" align=\"0\" show=\"1\"/>\n\t\t\t<up x=\"14\" y=\"27\" width=\"106\" align=\"0\" show=\"1\"/>\n\t\t\t<down x=\"122\" y=\"27\" width=\"106\" align=\"0\" show=\"1\"/>\n\t\t</layout_s>\n\n\t</layout>\n\t<preview width=\"250\" height=\"170\">\n\t\t<l x=\"0\" y=\"0\"/>\n\t\t<s x=\"0\" y=\"112\"/>\n\t</preview>\n</root>"
  },
  {
    "path": "TrafficMonitor/skins/皮肤01/skin.ini",
    "content": "[skin]\ntext_color=3803808\nskin_author=zy"
  },
  {
    "path": "TrafficMonitor/skins/皮肤02/skin.ini",
    "content": "[skin]\ntext_color=9844741\nskin_author=zy"
  },
  {
    "path": "TrafficMonitor/skins/皮肤03/skin.ini",
    "content": "[skin]\ntext_color=16777215\nskin_author=zy"
  },
  {
    "path": "TrafficMonitor/skins/皮肤04/skin.ini",
    "content": "[skin]\ntext_color=16777215\nskin_author=zy"
  },
  {
    "path": "TrafficMonitor/skins/皮肤05/skin.ini",
    "content": "[skin]\ntext_color=335674\nskin_author=zy"
  },
  {
    "path": "TrafficMonitor/skins/皮肤06/skin.ini",
    "content": "[skin]\ntext_color=16773870\nskin_author=zy"
  },
  {
    "path": "TrafficMonitor/skins/皮肤07/skin.ini",
    "content": "[skin]\ntext_color=14547455\nskin_author=zy"
  },
  {
    "path": "TrafficMonitor/skins/皮肤08/skin.ini",
    "content": "[skin]\ntext_color=16763528\nskin_author=zy"
  },
  {
    "path": "TrafficMonitor/skins/皮肤09/skin.ini",
    "content": "[skin]\ntext_color=16384\nskin_author=zy"
  },
  {
    "path": "TrafficMonitor/skins/皮肤10/skin.ini",
    "content": "[skin]\ntext_color=9586944\nskin_author=\"zy\"\n[layout]\ntext_height=20\nno_text=1\npreview_width=238\npreview_height=110\nwidth_l=189\nheight_l=43\nup_x_l=24\nup_y_l=2\nup_width_l=73\nup_align_l=0\ndown_x_l=116\ndown_y_l=2\ndown_width_l=73\ndown_align_l=0\ncpu_x_l=24\ncpu_y_l=21\ncpu_width_l=73\ncpu_align_l=0\nmemory_x_l=116\nmemory_y_l=21\nmemory_width_l=73\nmemory_align_l=0\nshow_up_l=1\nshow_down_l=1\nshow_cpu_l=1\nshow_memory_l=1\npreview_x_l=0\npreview_y_l=47\nwidth_s=189\nheight_s=28\nup_x_s=24\nup_y_s=4\nup_width_s=73\nup_align_s=0\ndown_x_s=116\ndown_y_s=4\ndown_width_s=73\ndown_align_s=0\ncpu_x_s=0\ncpu_y_s=0\ncpu_width_s=0\ncpu_align_s=0\nmemory_x_s=0\nmemory_y_s=0\nmemory_width_s=0\nmemory_align_s=0\nshow_up_s=1\nshow_down_s=1\nshow_cpu_s=0\nshow_memory_s=0\npreview_x_s=0\npreview_y_s=0\n"
  },
  {
    "path": "TrafficMonitor/skins/皮肤10（竖排）/skin.ini",
    "content": "[skin]\ntext_color=9586944\nskin_author=\"zy\"\n[layout]\ntext_height=20\nno_text=1\npreview_width=238\npreview_height=79\nwidth_l=110\nheight_l=63\nup_x_l=24\nup_y_l=2\nup_width_l=85\ndown_x_l=24\ndown_y_l=21\ndown_width_l=85\ncpu_x_l=24\ncpu_y_l=40\ncpu_width_l=34\nmemory_x_l=76\nmemory_y_l=40\nmemory_width_l=33\nshow_up_l=1\nshow_down_l=1\nshow_cpu_l=1\nshow_memory_l=1\npreview_x_l=115\npreview_y_l=0\nwidth_s=100\nheight_s=43\nup_x_s=24\nup_y_s=2\nup_width_s=76\ndown_x_s=24\ndown_y_s=20\ndown_width_s=76\ncpu_x_s=0\ncpu_y_s=0\ncpu_width_s=0\nmemory_x_s=0\nmemory_y_s=0\nmemory_width_s=0\nshow_up_s=1\nshow_down_s=1\nshow_cpu_s=0\nshow_memory_s=0\npreview_x_s=0\npreview_y_s=0\n"
  },
  {
    "path": "TrafficMonitor/skins/皮肤11/skin.ini",
    "content": "[skin]\ntext_color=16777215,0,0,0,\nspecify_each_item_color=0\nskin_author=\"zy\"\nfont_name=\"Segoe UI Semibold\"\nfont_size=10\nfont_style=0\n[layout]\ntext_height=20\nno_text=1\npreview_width=171\npreview_height=108\nwidth_l=174\nheight_l=41\nup_x_l=16\nup_y_l=1\nup_width_l=69\nup_align_l=1\ndown_x_l=16\ndown_y_l=20\ndown_width_l=69\ndown_align_l=1\ncpu_x_l=90\ncpu_y_l=20\ncpu_width_l=42\ncpu_align_l=2\nmemory_x_l=132\nmemory_y_l=20\nmemory_width_l=42\nmemory_align_l=2\nshow_up_l=1\nshow_down_l=1\nshow_cpu_l=1\nshow_memory_l=1\npreview_x_l=0\npreview_y_l=55\nwidth_s=90\nheight_s=41\nup_x_s=16\nup_y_s=1\nup_width_s=69\nup_align_s=1\ndown_x_s=16\ndown_y_s=20\ndown_width_s=69\ndown_align_s=1\ncpu_x_s=0\ncpu_y_s=0\ncpu_width_s=0\ncpu_align_s=0\nmemory_x_s=0\nmemory_y_s=0\nmemory_width_s=0\nmemory_align_s=0\nshow_up_s=1\nshow_down_s=1\nshow_cpu_s=0\nshow_memory_s=0\npreview_x_s=0\npreview_y_s=0\n"
  },
  {
    "path": "TrafficMonitor/skins/默认皮肤2/skin.ini",
    "content": "[skin]\ntext_color=16384\nskin_author=zy\n[layout]\ntext_height=20\nwidth_l=220\nheight_l=43\nup_x_l=6\nup_y_l=2\nup_width_l=108\ndown_x_l=114\ndown_y_l=2\ndown_width_l=110\ncpu_x_l=6\ncpu_y_l=21\ncpu_width_l=108\nmemory_x_l=114\nmemory_y_l=21\nmemory_width_l=110\nshow_up_l = 1\nshow_down_l = 1\nshow_cpu_l = 1\nshow_memory_l = 1\n\nwidth_s=153\nheight_s=28\ncpu_x_s=6\ncpu_y_s=4\ncpu_width_s=70\nmemory_x_s=77\nmemory_y_s=4\nmemory_width_s=77\nshow_item_s = 12\nshow_up_s = 0\nshow_down_s = 0\nshow_cpu_s = 1\nshow_memory_s = 1\n"
  },
  {
    "path": "TrafficMonitor/stdafx.cpp",
    "content": "\n// stdafx.cpp : ֻ׼ļԴļ\n// TrafficMonitor.pch ΪԤͷ\n// stdafx.obj ԤϢ\n\n#include \"stdafx.h\"\n\n\n"
  },
  {
    "path": "TrafficMonitor/stdafx.h",
    "content": "﻿\n// stdafx.h : 标准系统包含文件的包含文件，\n// 或是经常使用但不常更改的\n// 特定于项目的包含文件\n\n#pragma once\n\n#ifndef VC_EXTRALEAN\n#define VC_EXTRALEAN            // 从 Windows 头中排除极少使用的资料\n#endif\n\n#include \"targetver.h\"\n\n#define _ATL_CSTRING_EXPLICIT_CONSTRUCTORS      // 某些 CString 构造函数将是显式的\n\n// 关闭 MFC 对某些常见但经常可放心忽略的警告消息的隐藏\n#define _AFX_ALL_WARNINGS\n\n#include <afxwin.h>         // MFC 核心组件和标准组件\n#include <afxext.h>         // MFC 扩展\n\n\n#include <afxdisp.h>        // MFC 自动化类\n\n\n\n#ifndef _AFX_NO_OLE_SUPPORT\n#include <afxdtctl.h>           // MFC 对 Internet Explorer 4 公共控件的支持\n#endif\n#ifndef _AFX_NO_AFXCMN_SUPPORT\n#include <afxcmn.h>             // MFC 对 Windows 公共控件的支持\n#endif // _AFX_NO_AFXCMN_SUPPORT\n\n#include <afxcontrolbars.h>     // 功能区和控件条的 MFC 支持\n\n#include \"resource.h\"\n#include <string>\nusing std::string;\nusing std::wstring;\n#include <vector>\nusing std::vector;\n#include <deque>\nusing std::deque;\n#include<iostream>\n#include<fstream>\nusing std::ifstream;\nusing std::ofstream;\n#include <map>\n#include <set>\n#include<io.h>\n#include<algorithm>\n#include <cmath>\n#include <afxinet.h>    //用于支持使用网络相关的类\n#include <afxwin.h>\n\n\n#ifdef _UNICODE\n#if defined _M_IX86\n#pragma comment(linker,\"/manifestdependency:\\\"type='win32' name='Microsoft.Windows.Common-Controls' version='6.0.0.0' processorArchitecture='x86' publicKeyToken='6595b64144ccf1df' language='*'\\\"\")\n#elif defined _M_X64\n#pragma comment(linker,\"/manifestdependency:\\\"type='win32' name='Microsoft.Windows.Common-Controls' version='6.0.0.0' processorArchitecture='amd64' publicKeyToken='6595b64144ccf1df' language='*'\\\"\")\n#else\n#pragma comment(linker,\"/manifestdependency:\\\"type='win32' name='Microsoft.Windows.Common-Controls' version='6.0.0.0' processorArchitecture='*' publicKeyToken='6595b64144ccf1df' language='*'\\\"\")\n#endif\n#endif\n\n\n#define MY_WM_NOTIFYICON (WM_USER+1005)\n#define WM_TASKBAR_WND_CLOSED (WM_USER+1006)        //任务栏窗口关闭时发送此消息\n#define WM_MONITOR_INFO_UPDATED (WM_USER+1007)        //监控信息已更新\n#define WM_REOPEN_TASKBAR_WND (WM_USER+1008)        //重新打开任务栏窗口\n#define WM_SETTINGS_APPLIED (WM_USER+1009)          //在选项设置中点击了“应用”按钮\n\n#define WM_NEXT_USER_MSG (WM_USER+1010)\n\n//#define CONFIG_PATH _T(\".\\\\config.ini\")\n//#define CONFIG_PATHA \".\\\\config.ini\"\n//#define LOG_PATH _T(\".\\\\error.log\")\n//#define HISTORY_TRAFFIC_PATH _T(\".\\\\history_traffic.dat\")\n#define MAX_RESTART_CNT 10      //最大重新初始化次数\n\n#define MAIN_TIMER 1234         //主定时器的ID\n#define DELAY_TIMER 1235        //延时定时器ID\n#define TASKBAR_TIMER 1236\n#define CONNECTION_DETAIL_TIMER 1237\n#define MONITOR_TIMER 1238\n\n#define MAX_INSERT_TO_TASKBAR_CNT 5     //尝试嵌入任务栏的最大次数\n\n#define APP_NAME _T(\"TrafficMonitor\")\n#define APP_CLASS_NAME _T(\"TrafficMonitor_r7XZaS4p\") //程序主窗口的类名\n#define VERSION L\"1.83\"\n\n#define MAX_NOTIFY_ICON 6       //可选的通知区图标数量\n\n//背景图片的文件名\n#define BACKGROUND_IMAGE_S L\"\\\\background.bmp\"\n#define BACKGROUND_IMAGE_L L\"\\\\background_l.bmp\"\n#define BACKGROUND_MASK_S L\"\\\\background_mask.bmp\"\n#define BACKGROUND_MASK_L L\"\\\\background_mask_l.bmp\"\n\n#define MIN_FONT_SIZE 5\n#define MAX_FONT_SIZE 72\n\n#define SAFE_DELETE(p) do \\\n{\\\n    if(p != nullptr) \\\n    { \\\n        delete p; \\\n        p = nullptr; \\\n    } \\\n} while (false)\n\n//定义两个用于作为任务栏透明色的颜色（当需要设置任务栏窗口不透明时使用）\n#define TASKBAR_TRANSPARENT_COLOR1 RGB(52, 28, 41)\n#define TASKBAR_TRANSPARENT_COLOR2 RGB(38, 67, 55)\n\n//如果需要为Windows XP系统编译，请去掉下面一行代码的注释\n//#define COMPILE_FOR_WINXP\n"
  },
  {
    "path": "TrafficMonitor/targetver.h",
    "content": "#pragma once\n\n//  SDKDDKVer.h õ߰汾 Windows ƽ̨\n\n// ҪΪǰ Windows ƽ̨Ӧó WinSDKVer.h\n//  _WIN32_WINNT ΪҪֵ֧ƽ̨Ȼٰ SDKDDKVer.h\n\n#include <SDKDDKVer.h>\n"
  },
  {
    "path": "TrafficMonitor/tinyxml2/tinyxml2.cpp",
    "content": "﻿/*\nOriginal code by Lee Thomason (www.grinninglizard.com)\n\nThis software is provided 'as-is', without any express or implied\nwarranty. In no event will the authors be held liable for any\ndamages arising from the use of this software.\n\nPermission is granted to anyone to use this software for any\npurpose, including commercial applications, and to alter it and\nredistribute it freely, subject to the following restrictions:\n\n1. The origin of this software must not be misrepresented; you must\nnot claim that you wrote the original software. If you use this\nsoftware in a product, an acknowledgment in the product documentation\nwould be appreciated but is not required.\n\n2. Altered source versions must be plainly marked as such, and\nmust not be misrepresented as being the original software.\n\n3. This notice may not be removed or altered from any source\ndistribution.\n*/\n\n#include \"tinyxml2.h\"\n\n#include <new>\t\t// yes, this one new style header, is in the Android SDK.\n#if defined(ANDROID_NDK) || defined(__BORLANDC__) || defined(__QNXNTO__)\n#   include <stddef.h>\n#   include <stdarg.h>\n#else\n#   include <cstddef>\n#   include <cstdarg>\n#endif\n\n#if defined(_MSC_VER) && (_MSC_VER >= 1400 ) && (!defined WINCE)\n\t// Microsoft Visual Studio, version 2005 and higher. Not WinCE.\n\t/*int _snprintf_s(\n\t   char *buffer,\n\t   size_t sizeOfBuffer,\n\t   size_t count,\n\t   const char *format [,\n\t\t  argument] ...\n\t);*/\n\tstatic inline int TIXML_SNPRINTF( char* buffer, size_t size, const char* format, ... )\n\t{\n\t\tva_list va;\n\t\tva_start( va, format );\n\t\tconst int result = vsnprintf_s( buffer, size, _TRUNCATE, format, va );\n\t\tva_end( va );\n\t\treturn result;\n\t}\n\n\tstatic inline int TIXML_VSNPRINTF( char* buffer, size_t size, const char* format, va_list va )\n\t{\n\t\tconst int result = vsnprintf_s( buffer, size, _TRUNCATE, format, va );\n\t\treturn result;\n\t}\n\n\t#define TIXML_VSCPRINTF\t_vscprintf\n\t#define TIXML_SSCANF\tsscanf_s\n#elif defined _MSC_VER\n\t// Microsoft Visual Studio 2003 and earlier or WinCE\n\t#define TIXML_SNPRINTF\t_snprintf\n\t#define TIXML_VSNPRINTF _vsnprintf\n\t#define TIXML_SSCANF\tsscanf\n\t#if (_MSC_VER < 1400 ) && (!defined WINCE)\n\t\t// Microsoft Visual Studio 2003 and not WinCE.\n\t\t#define TIXML_VSCPRINTF   _vscprintf // VS2003's C runtime has this, but VC6 C runtime or WinCE SDK doesn't have.\n\t#else\n\t\t// Microsoft Visual Studio 2003 and earlier or WinCE.\n\t\tstatic inline int TIXML_VSCPRINTF( const char* format, va_list va )\n\t\t{\n\t\t\tint len = 512;\n\t\t\tfor (;;) {\n\t\t\t\tlen = len*2;\n\t\t\t\tchar* str = new char[len]();\n\t\t\t\tconst int required = _vsnprintf(str, len, format, va);\n\t\t\t\tdelete[] str;\n\t\t\t\tif ( required != -1 ) {\n\t\t\t\t\tTIXMLASSERT( required >= 0 );\n\t\t\t\t\tlen = required;\n\t\t\t\t\tbreak;\n\t\t\t\t}\n\t\t\t}\n\t\t\tTIXMLASSERT( len >= 0 );\n\t\t\treturn len;\n\t\t}\n\t#endif\n#else\n\t// GCC version 3 and higher\n\t//#warning( \"Using sn* functions.\" )\n\t#define TIXML_SNPRINTF\tsnprintf\n\t#define TIXML_VSNPRINTF\tvsnprintf\n\tstatic inline int TIXML_VSCPRINTF( const char* format, va_list va )\n\t{\n\t\tint len = vsnprintf( 0, 0, format, va );\n\t\tTIXMLASSERT( len >= 0 );\n\t\treturn len;\n\t}\n\t#define TIXML_SSCANF   sscanf\n#endif\n\n#if defined(_WIN64)\n\t#define TIXML_FSEEK _fseeki64\n\t#define TIXML_FTELL _ftelli64\n#elif defined(__APPLE__) || (__FreeBSD__)\n\t#define TIXML_FSEEK fseeko\n\t#define TIXML_FTELL ftello\n#elif defined(__unix__) && defined(__x86_64__)\n\t#define TIXML_FSEEK fseeko64\n\t#define TIXML_FTELL ftello64\n#else\n\t#define TIXML_FSEEK fseek\n\t#define TIXML_FTELL ftell\n#endif\n\n\nstatic const char LINE_FEED\t\t\t\t= static_cast<char>(0x0a);\t\t\t// all line endings are normalized to LF\nstatic const char LF = LINE_FEED;\nstatic const char CARRIAGE_RETURN\t\t= static_cast<char>(0x0d);\t\t\t// CR gets filtered out\nstatic const char CR = CARRIAGE_RETURN;\nstatic const char SINGLE_QUOTE\t\t\t= '\\'';\nstatic const char DOUBLE_QUOTE\t\t\t= '\\\"';\n\n// Bunch of unicode info at:\n//\t\thttp://www.unicode.org/faq/utf_bom.html\n//\tef bb bf (Microsoft \"lead bytes\") - designates UTF-8\n\nstatic const unsigned char TIXML_UTF_LEAD_0 = 0xefU;\nstatic const unsigned char TIXML_UTF_LEAD_1 = 0xbbU;\nstatic const unsigned char TIXML_UTF_LEAD_2 = 0xbfU;\n\nnamespace tinyxml2\n{\n\nstruct Entity {\n    const char* pattern;\n    int length;\n    char value;\n};\n\nstatic const int NUM_ENTITIES = 5;\nstatic const Entity entities[NUM_ENTITIES] = {\n    { \"quot\", 4,\tDOUBLE_QUOTE },\n    { \"amp\", 3,\t\t'&'  },\n    { \"apos\", 4,\tSINGLE_QUOTE },\n    { \"lt\",\t2, \t\t'<'\t },\n    { \"gt\",\t2,\t\t'>'\t }\n};\n\n\nStrPair::~StrPair()\n{\n    Reset();\n}\n\n\nvoid StrPair::TransferTo( StrPair* other )\n{\n    if ( this == other ) {\n        return;\n    }\n    // This in effect implements the assignment operator by \"moving\"\n    // ownership (as in auto_ptr).\n\n    TIXMLASSERT( other != 0 );\n    TIXMLASSERT( other->_flags == 0 );\n    TIXMLASSERT( other->_start == 0 );\n    TIXMLASSERT( other->_end == 0 );\n\n    other->Reset();\n\n    other->_flags = _flags;\n    other->_start = _start;\n    other->_end = _end;\n\n    _flags = 0;\n    _start = 0;\n    _end = 0;\n}\n\n\nvoid StrPair::Reset()\n{\n    if ( _flags & NEEDS_DELETE ) {\n        delete [] _start;\n    }\n    _flags = 0;\n    _start = 0;\n    _end = 0;\n}\n\n\nvoid StrPair::SetStr( const char* str, int flags )\n{\n    TIXMLASSERT( str );\n    Reset();\n    size_t len = strlen( str );\n    TIXMLASSERT( _start == 0 );\n    _start = new char[ len+1 ];\n    memcpy( _start, str, len+1 );\n    _end = _start + len;\n    _flags = flags | NEEDS_DELETE;\n}\n\n\nchar* StrPair::ParseText( char* p, const char* endTag, int strFlags, int* curLineNumPtr )\n{\n    TIXMLASSERT( p );\n    TIXMLASSERT( endTag && *endTag );\n\tTIXMLASSERT(curLineNumPtr);\n\n    char* start = p;\n    const char  endChar = *endTag;\n    size_t length = strlen( endTag );\n\n    // Inner loop of text parsing.\n    while ( *p ) {\n        if ( *p == endChar && strncmp( p, endTag, length ) == 0 ) {\n            Set( start, p, strFlags );\n            return p + length;\n        } else if (*p == '\\n') {\n            ++(*curLineNumPtr);\n        }\n        ++p;\n        TIXMLASSERT( p );\n    }\n    return 0;\n}\n\n\nchar* StrPair::ParseName( char* p )\n{\n    if ( !p || !(*p) ) {\n        return 0;\n    }\n    if ( !XMLUtil::IsNameStartChar( (unsigned char) *p ) ) {\n        return 0;\n    }\n\n    char* const start = p;\n    ++p;\n    while ( *p && XMLUtil::IsNameChar( (unsigned char) *p ) ) {\n        ++p;\n    }\n\n    Set( start, p, 0 );\n    return p;\n}\n\n\nvoid StrPair::CollapseWhitespace()\n{\n    // Adjusting _start would cause undefined behavior on delete[]\n    TIXMLASSERT( ( _flags & NEEDS_DELETE ) == 0 );\n    // Trim leading space.\n    _start = XMLUtil::SkipWhiteSpace( _start, 0 );\n\n    if ( *_start ) {\n        const char* p = _start;\t// the read pointer\n        char* q = _start;\t// the write pointer\n\n        while( *p ) {\n            if ( XMLUtil::IsWhiteSpace( *p )) {\n                p = XMLUtil::SkipWhiteSpace( p, 0 );\n                if ( *p == 0 ) {\n                    break;    // don't write to q; this trims the trailing space.\n                }\n                *q = ' ';\n                ++q;\n            }\n            *q = *p;\n            ++q;\n            ++p;\n        }\n        *q = 0;\n    }\n}\n\n\nconst char* StrPair::GetStr()\n{\n    TIXMLASSERT( _start );\n    TIXMLASSERT( _end );\n    if ( _flags & NEEDS_FLUSH ) {\n        *_end = 0;\n        _flags ^= NEEDS_FLUSH;\n\n        if ( _flags ) {\n            const char* p = _start;\t// the read pointer\n            char* q = _start;\t// the write pointer\n\n            while( p < _end ) {\n                if ( (_flags & NEEDS_NEWLINE_NORMALIZATION) && *p == CR ) {\n                    // CR-LF pair becomes LF\n                    // CR alone becomes LF\n                    // LF-CR becomes LF\n                    if ( *(p+1) == LF ) {\n                        p += 2;\n                    }\n                    else {\n                        ++p;\n                    }\n                    *q = LF;\n                    ++q;\n                }\n                else if ( (_flags & NEEDS_NEWLINE_NORMALIZATION) && *p == LF ) {\n                    if ( *(p+1) == CR ) {\n                        p += 2;\n                    }\n                    else {\n                        ++p;\n                    }\n                    *q = LF;\n                    ++q;\n                }\n                else if ( (_flags & NEEDS_ENTITY_PROCESSING) && *p == '&' ) {\n                    // Entities handled by tinyXML2:\n                    // - special entities in the entity table [in/out]\n                    // - numeric character reference [in]\n                    //   &#20013; or &#x4e2d;\n\n                    if ( *(p+1) == '#' ) {\n                        const int buflen = 10;\n                        char buf[buflen] = { 0 };\n                        int len = 0;\n                        const char* adjusted = const_cast<char*>( XMLUtil::GetCharacterRef( p, buf, &len ) );\n                        if ( adjusted == 0 ) {\n                            *q = *p;\n                            ++p;\n                            ++q;\n                        }\n                        else {\n                            TIXMLASSERT( 0 <= len && len <= buflen );\n                            TIXMLASSERT( q + len <= adjusted );\n                            p = adjusted;\n                            memcpy( q, buf, len );\n                            q += len;\n                        }\n                    }\n                    else {\n                        bool entityFound = false;\n                        for( int i = 0; i < NUM_ENTITIES; ++i ) {\n                            const Entity& entity = entities[i];\n                            if ( strncmp( p + 1, entity.pattern, entity.length ) == 0\n                                    && *( p + entity.length + 1 ) == ';' ) {\n                                // Found an entity - convert.\n                                *q = entity.value;\n                                ++q;\n                                p += entity.length + 2;\n                                entityFound = true;\n                                break;\n                            }\n                        }\n                        if ( !entityFound ) {\n                            // fixme: treat as error?\n                            ++p;\n                            ++q;\n                        }\n                    }\n                }\n                else {\n                    *q = *p;\n                    ++p;\n                    ++q;\n                }\n            }\n            *q = 0;\n        }\n        // The loop below has plenty going on, and this\n        // is a less useful mode. Break it out.\n        if ( _flags & NEEDS_WHITESPACE_COLLAPSING ) {\n            CollapseWhitespace();\n        }\n        _flags = (_flags & NEEDS_DELETE);\n    }\n    TIXMLASSERT( _start );\n    return _start;\n}\n\n\n\n\n// --------- XMLUtil ----------- //\n\nconst char* XMLUtil::writeBoolTrue  = \"true\";\nconst char* XMLUtil::writeBoolFalse = \"false\";\n\nvoid XMLUtil::SetBoolSerialization(const char* writeTrue, const char* writeFalse)\n{\n\tstatic const char* defTrue  = \"true\";\n\tstatic const char* defFalse = \"false\";\n\n\twriteBoolTrue = (writeTrue) ? writeTrue : defTrue;\n\twriteBoolFalse = (writeFalse) ? writeFalse : defFalse;\n}\n\n\nconst char* XMLUtil::ReadBOM( const char* p, bool* bom )\n{\n    TIXMLASSERT( p );\n    TIXMLASSERT( bom );\n    *bom = false;\n    const unsigned char* pu = reinterpret_cast<const unsigned char*>(p);\n    // Check for BOM:\n    if (    *(pu+0) == TIXML_UTF_LEAD_0\n            && *(pu+1) == TIXML_UTF_LEAD_1\n            && *(pu+2) == TIXML_UTF_LEAD_2 ) {\n        *bom = true;\n        p += 3;\n    }\n    TIXMLASSERT( p );\n    return p;\n}\n\n\nvoid XMLUtil::ConvertUTF32ToUTF8( unsigned long input, char* output, int* length )\n{\n    const unsigned long BYTE_MASK = 0xBF;\n    const unsigned long BYTE_MARK = 0x80;\n    const unsigned long FIRST_BYTE_MARK[7] = { 0x00, 0x00, 0xC0, 0xE0, 0xF0, 0xF8, 0xFC };\n\n    if (input < 0x80) {\n        *length = 1;\n    }\n    else if ( input < 0x800 ) {\n        *length = 2;\n    }\n    else if ( input < 0x10000 ) {\n        *length = 3;\n    }\n    else if ( input < 0x200000 ) {\n        *length = 4;\n    }\n    else {\n        *length = 0;    // This code won't convert this correctly anyway.\n        return;\n    }\n\n    output += *length;\n\n    // Scary scary fall throughs are annotated with carefully designed comments\n    // to suppress compiler warnings such as -Wimplicit-fallthrough in gcc\n    switch (*length) {\n        case 4:\n            --output;\n            *output = static_cast<char>((input | BYTE_MARK) & BYTE_MASK);\n            input >>= 6;\n            //fall through\n        case 3:\n            --output;\n            *output = static_cast<char>((input | BYTE_MARK) & BYTE_MASK);\n            input >>= 6;\n            //fall through\n        case 2:\n            --output;\n            *output = static_cast<char>((input | BYTE_MARK) & BYTE_MASK);\n            input >>= 6;\n            //fall through\n        case 1:\n            --output;\n            *output = static_cast<char>(input | FIRST_BYTE_MARK[*length]);\n            break;\n        default:\n            TIXMLASSERT( false );\n    }\n}\n\n\nconst char* XMLUtil::GetCharacterRef( const char* p, char* value, int* length )\n{\n    // Presume an entity, and pull it out.\n    *length = 0;\n\n    if ( *(p+1) == '#' && *(p+2) ) {\n        unsigned long ucs = 0;\n        TIXMLASSERT( sizeof( ucs ) >= 4 );\n        ptrdiff_t delta = 0;\n        unsigned mult = 1;\n        static const char SEMICOLON = ';';\n\n        if ( *(p+2) == 'x' ) {\n            // Hexadecimal.\n            const char* q = p+3;\n            if ( !(*q) ) {\n                return 0;\n            }\n\n            q = strchr( q, SEMICOLON );\n\n            if ( !q ) {\n                return 0;\n            }\n            TIXMLASSERT( *q == SEMICOLON );\n\n            delta = q-p;\n            --q;\n\n            while ( *q != 'x' ) {\n                unsigned int digit = 0;\n\n                if ( *q >= '0' && *q <= '9' ) {\n                    digit = *q - '0';\n                }\n                else if ( *q >= 'a' && *q <= 'f' ) {\n                    digit = *q - 'a' + 10;\n                }\n                else if ( *q >= 'A' && *q <= 'F' ) {\n                    digit = *q - 'A' + 10;\n                }\n                else {\n                    return 0;\n                }\n                TIXMLASSERT( digit < 16 );\n                TIXMLASSERT( digit == 0 || mult <= UINT_MAX / digit );\n                const unsigned int digitScaled = mult * digit;\n                TIXMLASSERT( ucs <= ULONG_MAX - digitScaled );\n                ucs += digitScaled;\n                TIXMLASSERT( mult <= UINT_MAX / 16 );\n                mult *= 16;\n                --q;\n            }\n        }\n        else {\n            // Decimal.\n            const char* q = p+2;\n            if ( !(*q) ) {\n                return 0;\n            }\n\n            q = strchr( q, SEMICOLON );\n\n            if ( !q ) {\n                return 0;\n            }\n            TIXMLASSERT( *q == SEMICOLON );\n\n            delta = q-p;\n            --q;\n\n            while ( *q != '#' ) {\n                if ( *q >= '0' && *q <= '9' ) {\n                    const unsigned int digit = *q - '0';\n                    TIXMLASSERT( digit < 10 );\n                    TIXMLASSERT( digit == 0 || mult <= UINT_MAX / digit );\n                    const unsigned int digitScaled = mult * digit;\n                    TIXMLASSERT( ucs <= ULONG_MAX - digitScaled );\n                    ucs += digitScaled;\n                }\n                else {\n                    return 0;\n                }\n                TIXMLASSERT( mult <= UINT_MAX / 10 );\n                mult *= 10;\n                --q;\n            }\n        }\n        // convert the UCS to UTF-8\n        ConvertUTF32ToUTF8( ucs, value, length );\n        return p + delta + 1;\n    }\n    return p+1;\n}\n\n\nvoid XMLUtil::ToStr( int v, char* buffer, int bufferSize )\n{\n    TIXML_SNPRINTF( buffer, bufferSize, \"%d\", v );\n}\n\n\nvoid XMLUtil::ToStr( unsigned v, char* buffer, int bufferSize )\n{\n    TIXML_SNPRINTF( buffer, bufferSize, \"%u\", v );\n}\n\n\nvoid XMLUtil::ToStr( bool v, char* buffer, int bufferSize )\n{\n    TIXML_SNPRINTF( buffer, bufferSize, \"%s\", v ? writeBoolTrue : writeBoolFalse);\n}\n\n/*\n\tToStr() of a number is a very tricky topic.\n\thttps://github.com/leethomason/tinyxml2/issues/106\n*/\nvoid XMLUtil::ToStr( float v, char* buffer, int bufferSize )\n{\n    TIXML_SNPRINTF( buffer, bufferSize, \"%.8g\", v );\n}\n\n\nvoid XMLUtil::ToStr( double v, char* buffer, int bufferSize )\n{\n    TIXML_SNPRINTF( buffer, bufferSize, \"%.17g\", v );\n}\n\n\nvoid XMLUtil::ToStr( int64_t v, char* buffer, int bufferSize )\n{\n\t// horrible syntax trick to make the compiler happy about %lld\n\tTIXML_SNPRINTF(buffer, bufferSize, \"%lld\", static_cast<long long>(v));\n}\n\nvoid XMLUtil::ToStr( uint64_t v, char* buffer, int bufferSize )\n{\n    // horrible syntax trick to make the compiler happy about %llu\n    TIXML_SNPRINTF(buffer, bufferSize, \"%llu\", (long long)v);\n}\n\nbool XMLUtil::ToInt(const char* str, int* value)\n{\n    if (TIXML_SSCANF(str, IsPrefixHex(str) ? \"%x\" : \"%d\", value) == 1) {\n        return true;\n    }\n    return false;\n}\n\nbool XMLUtil::ToUnsigned(const char* str, unsigned* value)\n{\n    if (TIXML_SSCANF(str, IsPrefixHex(str) ? \"%x\" : \"%u\", value) == 1) {\n        return true;\n    }\n    return false;\n}\n\nbool XMLUtil::ToBool( const char* str, bool* value )\n{\n    int ival = 0;\n    if ( ToInt( str, &ival )) {\n        *value = (ival==0) ? false : true;\n        return true;\n    }\n    static const char* TRUE_VALS[] = { \"true\", \"True\", \"TRUE\", 0 };\n    static const char* FALSE_VALS[] = { \"false\", \"False\", \"FALSE\", 0 };\n\n    for (int i = 0; TRUE_VALS[i]; ++i) {\n        if (StringEqual(str, TRUE_VALS[i])) {\n            *value = true;\n            return true;\n        }\n    }\n    for (int i = 0; FALSE_VALS[i]; ++i) {\n        if (StringEqual(str, FALSE_VALS[i])) {\n            *value = false;\n            return true;\n        }\n    }\n    return false;\n}\n\n\nbool XMLUtil::ToFloat( const char* str, float* value )\n{\n    if ( TIXML_SSCANF( str, \"%f\", value ) == 1 ) {\n        return true;\n    }\n    return false;\n}\n\n\nbool XMLUtil::ToDouble( const char* str, double* value )\n{\n    if ( TIXML_SSCANF( str, \"%lf\", value ) == 1 ) {\n        return true;\n    }\n    return false;\n}\n\n\nbool XMLUtil::ToInt64(const char* str, int64_t* value)\n{\n\tlong long v = 0;\t// horrible syntax trick to make the compiler happy about %lld\n\tif (TIXML_SSCANF(str, IsPrefixHex(str) ? \"%llx\" : \"%lld\", &v) == 1) {\n\t\t*value = static_cast<int64_t>(v);\n\t\treturn true;\n\t}\n\treturn false;\n}\n\n\nbool XMLUtil::ToUnsigned64(const char* str, uint64_t* value) {\n    unsigned long long v = 0;\t// horrible syntax trick to make the compiler happy about %llu\n    if(TIXML_SSCANF(str, IsPrefixHex(str) ? \"%llx\" : \"%llu\", &v) == 1) {\n        *value = (uint64_t)v;\n        return true;\n    }\n    return false;\n}\n\n\nchar* XMLDocument::Identify( char* p, XMLNode** node )\n{\n    TIXMLASSERT( node );\n    TIXMLASSERT( p );\n    char* const start = p;\n    int const startLine = _parseCurLineNum;\n    p = XMLUtil::SkipWhiteSpace( p, &_parseCurLineNum );\n    if( !*p ) {\n        *node = 0;\n        TIXMLASSERT( p );\n        return p;\n    }\n\n    // These strings define the matching patterns:\n    static const char* xmlHeader\t\t= { \"<?\" };\n    static const char* commentHeader\t= { \"<!--\" };\n    static const char* cdataHeader\t\t= { \"<![CDATA[\" };\n    static const char* dtdHeader\t\t= { \"<!\" };\n    static const char* elementHeader\t= { \"<\" };\t// and a header for everything else; check last.\n\n    static const int xmlHeaderLen\t\t= 2;\n    static const int commentHeaderLen\t= 4;\n    static const int cdataHeaderLen\t\t= 9;\n    static const int dtdHeaderLen\t\t= 2;\n    static const int elementHeaderLen\t= 1;\n\n    TIXMLASSERT( sizeof( XMLComment ) == sizeof( XMLUnknown ) );\t\t// use same memory pool\n    TIXMLASSERT( sizeof( XMLComment ) == sizeof( XMLDeclaration ) );\t// use same memory pool\n    XMLNode* returnNode = 0;\n    if ( XMLUtil::StringEqual( p, xmlHeader, xmlHeaderLen ) ) {\n        returnNode = CreateUnlinkedNode<XMLDeclaration>( _commentPool );\n        returnNode->_parseLineNum = _parseCurLineNum;\n        p += xmlHeaderLen;\n    }\n    else if ( XMLUtil::StringEqual( p, commentHeader, commentHeaderLen ) ) {\n        returnNode = CreateUnlinkedNode<XMLComment>( _commentPool );\n        returnNode->_parseLineNum = _parseCurLineNum;\n        p += commentHeaderLen;\n    }\n    else if ( XMLUtil::StringEqual( p, cdataHeader, cdataHeaderLen ) ) {\n        XMLText* text = CreateUnlinkedNode<XMLText>( _textPool );\n        returnNode = text;\n        returnNode->_parseLineNum = _parseCurLineNum;\n        p += cdataHeaderLen;\n        text->SetCData( true );\n    }\n    else if ( XMLUtil::StringEqual( p, dtdHeader, dtdHeaderLen ) ) {\n        returnNode = CreateUnlinkedNode<XMLUnknown>( _commentPool );\n        returnNode->_parseLineNum = _parseCurLineNum;\n        p += dtdHeaderLen;\n    }\n    else if ( XMLUtil::StringEqual( p, elementHeader, elementHeaderLen ) ) {\n        returnNode =  CreateUnlinkedNode<XMLElement>( _elementPool );\n        returnNode->_parseLineNum = _parseCurLineNum;\n        p += elementHeaderLen;\n    }\n    else {\n        returnNode = CreateUnlinkedNode<XMLText>( _textPool );\n        returnNode->_parseLineNum = _parseCurLineNum; // Report line of first non-whitespace character\n        p = start;\t// Back it up, all the text counts.\n        _parseCurLineNum = startLine;\n    }\n\n    TIXMLASSERT( returnNode );\n    TIXMLASSERT( p );\n    *node = returnNode;\n    return p;\n}\n\n\nbool XMLDocument::Accept( XMLVisitor* visitor ) const\n{\n    TIXMLASSERT( visitor );\n    if ( visitor->VisitEnter( *this ) ) {\n        for ( const XMLNode* node=FirstChild(); node; node=node->NextSibling() ) {\n            if ( !node->Accept( visitor ) ) {\n                break;\n            }\n        }\n    }\n    return visitor->VisitExit( *this );\n}\n\n\n// --------- XMLNode ----------- //\n\nXMLNode::XMLNode( XMLDocument* doc ) :\n    _document( doc ),\n    _parent( 0 ),\n    _value(),\n    _parseLineNum( 0 ),\n    _firstChild( 0 ), _lastChild( 0 ),\n    _prev( 0 ), _next( 0 ),\n\t_userData( 0 ),\n    _memPool( 0 )\n{\n}\n\n\nXMLNode::~XMLNode()\n{\n    DeleteChildren();\n    if ( _parent ) {\n        _parent->Unlink( this );\n    }\n}\n\nconst char* XMLNode::Value() const\n{\n    // Edge case: XMLDocuments don't have a Value. Return null.\n    if ( this->ToDocument() )\n        return 0;\n    return _value.GetStr();\n}\n\nvoid XMLNode::SetValue( const char* str, bool staticMem )\n{\n    if ( staticMem ) {\n        _value.SetInternedStr( str );\n    }\n    else {\n        _value.SetStr( str );\n    }\n}\n\nXMLNode* XMLNode::DeepClone(XMLDocument* target) const\n{\n\tXMLNode* clone = this->ShallowClone(target);\n\tif (!clone) return 0;\n\n\tfor (const XMLNode* child = this->FirstChild(); child; child = child->NextSibling()) {\n\t\tXMLNode* childClone = child->DeepClone(target);\n\t\tTIXMLASSERT(childClone);\n\t\tclone->InsertEndChild(childClone);\n\t}\n\treturn clone;\n}\n\nvoid XMLNode::DeleteChildren()\n{\n    while( _firstChild ) {\n        TIXMLASSERT( _lastChild );\n        DeleteChild( _firstChild );\n    }\n    _firstChild = _lastChild = 0;\n}\n\n\nvoid XMLNode::Unlink( XMLNode* child )\n{\n    TIXMLASSERT( child );\n    TIXMLASSERT( child->_document == _document );\n    TIXMLASSERT( child->_parent == this );\n    if ( child == _firstChild ) {\n        _firstChild = _firstChild->_next;\n    }\n    if ( child == _lastChild ) {\n        _lastChild = _lastChild->_prev;\n    }\n\n    if ( child->_prev ) {\n        child->_prev->_next = child->_next;\n    }\n    if ( child->_next ) {\n        child->_next->_prev = child->_prev;\n    }\n\tchild->_next = 0;\n\tchild->_prev = 0;\n\tchild->_parent = 0;\n}\n\n\nvoid XMLNode::DeleteChild( XMLNode* node )\n{\n    TIXMLASSERT( node );\n    TIXMLASSERT( node->_document == _document );\n    TIXMLASSERT( node->_parent == this );\n    Unlink( node );\n\tTIXMLASSERT(node->_prev == 0);\n\tTIXMLASSERT(node->_next == 0);\n\tTIXMLASSERT(node->_parent == 0);\n    DeleteNode( node );\n}\n\n\nXMLNode* XMLNode::InsertEndChild( XMLNode* addThis )\n{\n    TIXMLASSERT( addThis );\n    if ( addThis->_document != _document ) {\n        TIXMLASSERT( false );\n        return 0;\n    }\n    InsertChildPreamble( addThis );\n\n    if ( _lastChild ) {\n        TIXMLASSERT( _firstChild );\n        TIXMLASSERT( _lastChild->_next == 0 );\n        _lastChild->_next = addThis;\n        addThis->_prev = _lastChild;\n        _lastChild = addThis;\n\n        addThis->_next = 0;\n    }\n    else {\n        TIXMLASSERT( _firstChild == 0 );\n        _firstChild = _lastChild = addThis;\n\n        addThis->_prev = 0;\n        addThis->_next = 0;\n    }\n    addThis->_parent = this;\n    return addThis;\n}\n\n\nXMLNode* XMLNode::InsertFirstChild( XMLNode* addThis )\n{\n    TIXMLASSERT( addThis );\n    if ( addThis->_document != _document ) {\n        TIXMLASSERT( false );\n        return 0;\n    }\n    InsertChildPreamble( addThis );\n\n    if ( _firstChild ) {\n        TIXMLASSERT( _lastChild );\n        TIXMLASSERT( _firstChild->_prev == 0 );\n\n        _firstChild->_prev = addThis;\n        addThis->_next = _firstChild;\n        _firstChild = addThis;\n\n        addThis->_prev = 0;\n    }\n    else {\n        TIXMLASSERT( _lastChild == 0 );\n        _firstChild = _lastChild = addThis;\n\n        addThis->_prev = 0;\n        addThis->_next = 0;\n    }\n    addThis->_parent = this;\n    return addThis;\n}\n\n\nXMLNode* XMLNode::InsertAfterChild( XMLNode* afterThis, XMLNode* addThis )\n{\n    TIXMLASSERT( addThis );\n    if ( addThis->_document != _document ) {\n        TIXMLASSERT( false );\n        return 0;\n    }\n\n    TIXMLASSERT( afterThis );\n\n    if ( afterThis->_parent != this ) {\n        TIXMLASSERT( false );\n        return 0;\n    }\n    if ( afterThis == addThis ) {\n        // Current state: BeforeThis -> AddThis -> OneAfterAddThis\n        // Now AddThis must disappear from it's location and then\n        // reappear between BeforeThis and OneAfterAddThis.\n        // So just leave it where it is.\n        return addThis;\n    }\n\n    if ( afterThis->_next == 0 ) {\n        // The last node or the only node.\n        return InsertEndChild( addThis );\n    }\n    InsertChildPreamble( addThis );\n    addThis->_prev = afterThis;\n    addThis->_next = afterThis->_next;\n    afterThis->_next->_prev = addThis;\n    afterThis->_next = addThis;\n    addThis->_parent = this;\n    return addThis;\n}\n\n\n\n\nconst XMLElement* XMLNode::FirstChildElement( const char* name ) const\n{\n    for( const XMLNode* node = _firstChild; node; node = node->_next ) {\n        const XMLElement* element = node->ToElementWithName( name );\n        if ( element ) {\n            return element;\n        }\n    }\n    return 0;\n}\n\n\nconst XMLElement* XMLNode::LastChildElement( const char* name ) const\n{\n    for( const XMLNode* node = _lastChild; node; node = node->_prev ) {\n        const XMLElement* element = node->ToElementWithName( name );\n        if ( element ) {\n            return element;\n        }\n    }\n    return 0;\n}\n\n\nconst XMLElement* XMLNode::NextSiblingElement( const char* name ) const\n{\n    for( const XMLNode* node = _next; node; node = node->_next ) {\n        const XMLElement* element = node->ToElementWithName( name );\n        if ( element ) {\n            return element;\n        }\n    }\n    return 0;\n}\n\n\nconst XMLElement* XMLNode::PreviousSiblingElement( const char* name ) const\n{\n    for( const XMLNode* node = _prev; node; node = node->_prev ) {\n        const XMLElement* element = node->ToElementWithName( name );\n        if ( element ) {\n            return element;\n        }\n    }\n    return 0;\n}\n\n\nchar* XMLNode::ParseDeep( char* p, StrPair* parentEndTag, int* curLineNumPtr )\n{\n    // This is a recursive method, but thinking about it \"at the current level\"\n    // it is a pretty simple flat list:\n    //\t\t<foo/>\n    //\t\t<!-- comment -->\n    //\n    // With a special case:\n    //\t\t<foo>\n    //\t\t</foo>\n    //\t\t<!-- comment -->\n    //\n    // Where the closing element (/foo) *must* be the next thing after the opening\n    // element, and the names must match. BUT the tricky bit is that the closing\n    // element will be read by the child.\n    //\n    // 'endTag' is the end tag for this node, it is returned by a call to a child.\n    // 'parentEnd' is the end tag for the parent, which is filled in and returned.\n\n\tXMLDocument::DepthTracker tracker(_document);\n\tif (_document->Error())\n\t\treturn 0;\n\n\twhile( p && *p ) {\n        XMLNode* node = 0;\n\n        p = _document->Identify( p, &node );\n        TIXMLASSERT( p );\n        if ( node == 0 ) {\n            break;\n        }\n\n       const int initialLineNum = node->_parseLineNum;\n\n        StrPair endTag;\n        p = node->ParseDeep( p, &endTag, curLineNumPtr );\n        if ( !p ) {\n            DeleteNode( node );\n            if ( !_document->Error() ) {\n                _document->SetError( XML_ERROR_PARSING, initialLineNum, 0);\n            }\n            break;\n        }\n\n        const XMLDeclaration* const decl = node->ToDeclaration();\n        if ( decl ) {\n            // Declarations are only allowed at document level\n            //\n            // Multiple declarations are allowed but all declarations\n            // must occur before anything else.\n            //\n            // Optimized due to a security test case. If the first node is\n            // a declaration, and the last node is a declaration, then only\n            // declarations have so far been added.\n            bool wellLocated = false;\n\n            if (ToDocument()) {\n                if (FirstChild()) {\n                    wellLocated =\n                        FirstChild() &&\n                        FirstChild()->ToDeclaration() &&\n                        LastChild() &&\n                        LastChild()->ToDeclaration();\n                }\n                else {\n                    wellLocated = true;\n                }\n            }\n            if ( !wellLocated ) {\n                _document->SetError( XML_ERROR_PARSING_DECLARATION, initialLineNum, \"XMLDeclaration value=%s\", decl->Value());\n                DeleteNode( node );\n                break;\n            }\n        }\n\n        XMLElement* ele = node->ToElement();\n        if ( ele ) {\n            // We read the end tag. Return it to the parent.\n            if ( ele->ClosingType() == XMLElement::CLOSING ) {\n                if ( parentEndTag ) {\n                    ele->_value.TransferTo( parentEndTag );\n                }\n                node->_memPool->SetTracked();   // created and then immediately deleted.\n                DeleteNode( node );\n                return p;\n            }\n\n            // Handle an end tag returned to this level.\n            // And handle a bunch of annoying errors.\n            bool mismatch = false;\n            if ( endTag.Empty() ) {\n                if ( ele->ClosingType() == XMLElement::OPEN ) {\n                    mismatch = true;\n                }\n            }\n            else {\n                if ( ele->ClosingType() != XMLElement::OPEN ) {\n                    mismatch = true;\n                }\n                else if ( !XMLUtil::StringEqual( endTag.GetStr(), ele->Name() ) ) {\n                    mismatch = true;\n                }\n            }\n            if ( mismatch ) {\n                _document->SetError( XML_ERROR_MISMATCHED_ELEMENT, initialLineNum, \"XMLElement name=%s\", ele->Name());\n                DeleteNode( node );\n                break;\n            }\n        }\n        InsertEndChild( node );\n    }\n    return 0;\n}\n\n/*static*/ void XMLNode::DeleteNode( XMLNode* node )\n{\n    if ( node == 0 ) {\n        return;\n    }\n\tTIXMLASSERT(node->_document);\n\tif (!node->ToDocument()) {\n\t\tnode->_document->MarkInUse(node);\n\t}\n\n    MemPool* pool = node->_memPool;\n    node->~XMLNode();\n    pool->Free( node );\n}\n\nvoid XMLNode::InsertChildPreamble( XMLNode* insertThis ) const\n{\n    TIXMLASSERT( insertThis );\n    TIXMLASSERT( insertThis->_document == _document );\n\n\tif (insertThis->_parent) {\n        insertThis->_parent->Unlink( insertThis );\n\t}\n\telse {\n\t\tinsertThis->_document->MarkInUse(insertThis);\n        insertThis->_memPool->SetTracked();\n\t}\n}\n\nconst XMLElement* XMLNode::ToElementWithName( const char* name ) const\n{\n    const XMLElement* element = this->ToElement();\n    if ( element == 0 ) {\n        return 0;\n    }\n    if ( name == 0 ) {\n        return element;\n    }\n    if ( XMLUtil::StringEqual( element->Name(), name ) ) {\n       return element;\n    }\n    return 0;\n}\n\n// --------- XMLText ---------- //\nchar* XMLText::ParseDeep( char* p, StrPair*, int* curLineNumPtr )\n{\n    if ( this->CData() ) {\n        p = _value.ParseText( p, \"]]>\", StrPair::NEEDS_NEWLINE_NORMALIZATION, curLineNumPtr );\n        if ( !p ) {\n            _document->SetError( XML_ERROR_PARSING_CDATA, _parseLineNum, 0 );\n        }\n        return p;\n    }\n    else {\n        int flags = _document->ProcessEntities() ? StrPair::TEXT_ELEMENT : StrPair::TEXT_ELEMENT_LEAVE_ENTITIES;\n        if ( _document->WhitespaceMode() == COLLAPSE_WHITESPACE ) {\n            flags |= StrPair::NEEDS_WHITESPACE_COLLAPSING;\n        }\n\n        p = _value.ParseText( p, \"<\", flags, curLineNumPtr );\n        if ( p && *p ) {\n            return p-1;\n        }\n        if ( !p ) {\n            _document->SetError( XML_ERROR_PARSING_TEXT, _parseLineNum, 0 );\n        }\n    }\n    return 0;\n}\n\n\nXMLNode* XMLText::ShallowClone( XMLDocument* doc ) const\n{\n    if ( !doc ) {\n        doc = _document;\n    }\n    XMLText* text = doc->NewText( Value() );\t// fixme: this will always allocate memory. Intern?\n    text->SetCData( this->CData() );\n    return text;\n}\n\n\nbool XMLText::ShallowEqual( const XMLNode* compare ) const\n{\n    TIXMLASSERT( compare );\n    const XMLText* text = compare->ToText();\n    return ( text && XMLUtil::StringEqual( text->Value(), Value() ) );\n}\n\n\nbool XMLText::Accept( XMLVisitor* visitor ) const\n{\n    TIXMLASSERT( visitor );\n    return visitor->Visit( *this );\n}\n\n\n// --------- XMLComment ---------- //\n\nXMLComment::XMLComment( XMLDocument* doc ) : XMLNode( doc )\n{\n}\n\n\nXMLComment::~XMLComment()\n{\n}\n\n\nchar* XMLComment::ParseDeep( char* p, StrPair*, int* curLineNumPtr )\n{\n    // Comment parses as text.\n    p = _value.ParseText( p, \"-->\", StrPair::COMMENT, curLineNumPtr );\n    if ( p == 0 ) {\n        _document->SetError( XML_ERROR_PARSING_COMMENT, _parseLineNum, 0 );\n    }\n    return p;\n}\n\n\nXMLNode* XMLComment::ShallowClone( XMLDocument* doc ) const\n{\n    if ( !doc ) {\n        doc = _document;\n    }\n    XMLComment* comment = doc->NewComment( Value() );\t// fixme: this will always allocate memory. Intern?\n    return comment;\n}\n\n\nbool XMLComment::ShallowEqual( const XMLNode* compare ) const\n{\n    TIXMLASSERT( compare );\n    const XMLComment* comment = compare->ToComment();\n    return ( comment && XMLUtil::StringEqual( comment->Value(), Value() ));\n}\n\n\nbool XMLComment::Accept( XMLVisitor* visitor ) const\n{\n    TIXMLASSERT( visitor );\n    return visitor->Visit( *this );\n}\n\n\n// --------- XMLDeclaration ---------- //\n\nXMLDeclaration::XMLDeclaration( XMLDocument* doc ) : XMLNode( doc )\n{\n}\n\n\nXMLDeclaration::~XMLDeclaration()\n{\n    //printf( \"~XMLDeclaration\\n\" );\n}\n\n\nchar* XMLDeclaration::ParseDeep( char* p, StrPair*, int* curLineNumPtr )\n{\n    // Declaration parses as text.\n    p = _value.ParseText( p, \"?>\", StrPair::NEEDS_NEWLINE_NORMALIZATION, curLineNumPtr );\n    if ( p == 0 ) {\n        _document->SetError( XML_ERROR_PARSING_DECLARATION, _parseLineNum, 0 );\n    }\n    return p;\n}\n\n\nXMLNode* XMLDeclaration::ShallowClone( XMLDocument* doc ) const\n{\n    if ( !doc ) {\n        doc = _document;\n    }\n    XMLDeclaration* dec = doc->NewDeclaration( Value() );\t// fixme: this will always allocate memory. Intern?\n    return dec;\n}\n\n\nbool XMLDeclaration::ShallowEqual( const XMLNode* compare ) const\n{\n    TIXMLASSERT( compare );\n    const XMLDeclaration* declaration = compare->ToDeclaration();\n    return ( declaration && XMLUtil::StringEqual( declaration->Value(), Value() ));\n}\n\n\n\nbool XMLDeclaration::Accept( XMLVisitor* visitor ) const\n{\n    TIXMLASSERT( visitor );\n    return visitor->Visit( *this );\n}\n\n// --------- XMLUnknown ---------- //\n\nXMLUnknown::XMLUnknown( XMLDocument* doc ) : XMLNode( doc )\n{\n}\n\n\nXMLUnknown::~XMLUnknown()\n{\n}\n\n\nchar* XMLUnknown::ParseDeep( char* p, StrPair*, int* curLineNumPtr )\n{\n    // Unknown parses as text.\n    p = _value.ParseText( p, \">\", StrPair::NEEDS_NEWLINE_NORMALIZATION, curLineNumPtr );\n    if ( !p ) {\n        _document->SetError( XML_ERROR_PARSING_UNKNOWN, _parseLineNum, 0 );\n    }\n    return p;\n}\n\n\nXMLNode* XMLUnknown::ShallowClone( XMLDocument* doc ) const\n{\n    if ( !doc ) {\n        doc = _document;\n    }\n    XMLUnknown* text = doc->NewUnknown( Value() );\t// fixme: this will always allocate memory. Intern?\n    return text;\n}\n\n\nbool XMLUnknown::ShallowEqual( const XMLNode* compare ) const\n{\n    TIXMLASSERT( compare );\n    const XMLUnknown* unknown = compare->ToUnknown();\n    return ( unknown && XMLUtil::StringEqual( unknown->Value(), Value() ));\n}\n\n\nbool XMLUnknown::Accept( XMLVisitor* visitor ) const\n{\n    TIXMLASSERT( visitor );\n    return visitor->Visit( *this );\n}\n\n// --------- XMLAttribute ---------- //\n\nconst char* XMLAttribute::Name() const\n{\n    return _name.GetStr();\n}\n\nconst char* XMLAttribute::Value() const\n{\n    return _value.GetStr();\n}\n\nchar* XMLAttribute::ParseDeep( char* p, bool processEntities, int* curLineNumPtr )\n{\n    // Parse using the name rules: bug fix, was using ParseText before\n    p = _name.ParseName( p );\n    if ( !p || !*p ) {\n        return 0;\n    }\n\n    // Skip white space before =\n    p = XMLUtil::SkipWhiteSpace( p, curLineNumPtr );\n    if ( *p != '=' ) {\n        return 0;\n    }\n\n    ++p;\t// move up to opening quote\n    p = XMLUtil::SkipWhiteSpace( p, curLineNumPtr );\n    if ( *p != '\\\"' && *p != '\\'' ) {\n        return 0;\n    }\n\n    const char endTag[2] = { *p, 0 };\n    ++p;\t// move past opening quote\n\n    p = _value.ParseText( p, endTag, processEntities ? StrPair::ATTRIBUTE_VALUE : StrPair::ATTRIBUTE_VALUE_LEAVE_ENTITIES, curLineNumPtr );\n    return p;\n}\n\n\nvoid XMLAttribute::SetName( const char* n )\n{\n    _name.SetStr( n );\n}\n\n\nXMLError XMLAttribute::QueryIntValue( int* value ) const\n{\n    if ( XMLUtil::ToInt( Value(), value )) {\n        return XML_SUCCESS;\n    }\n    return XML_WRONG_ATTRIBUTE_TYPE;\n}\n\n\nXMLError XMLAttribute::QueryUnsignedValue( unsigned int* value ) const\n{\n    if ( XMLUtil::ToUnsigned( Value(), value )) {\n        return XML_SUCCESS;\n    }\n    return XML_WRONG_ATTRIBUTE_TYPE;\n}\n\n\nXMLError XMLAttribute::QueryInt64Value(int64_t* value) const\n{\n\tif (XMLUtil::ToInt64(Value(), value)) {\n\t\treturn XML_SUCCESS;\n\t}\n\treturn XML_WRONG_ATTRIBUTE_TYPE;\n}\n\n\nXMLError XMLAttribute::QueryUnsigned64Value(uint64_t* value) const\n{\n    if(XMLUtil::ToUnsigned64(Value(), value)) {\n        return XML_SUCCESS;\n    }\n    return XML_WRONG_ATTRIBUTE_TYPE;\n}\n\n\nXMLError XMLAttribute::QueryBoolValue( bool* value ) const\n{\n    if ( XMLUtil::ToBool( Value(), value )) {\n        return XML_SUCCESS;\n    }\n    return XML_WRONG_ATTRIBUTE_TYPE;\n}\n\n\nXMLError XMLAttribute::QueryFloatValue( float* value ) const\n{\n    if ( XMLUtil::ToFloat( Value(), value )) {\n        return XML_SUCCESS;\n    }\n    return XML_WRONG_ATTRIBUTE_TYPE;\n}\n\n\nXMLError XMLAttribute::QueryDoubleValue( double* value ) const\n{\n    if ( XMLUtil::ToDouble( Value(), value )) {\n        return XML_SUCCESS;\n    }\n    return XML_WRONG_ATTRIBUTE_TYPE;\n}\n\n\nvoid XMLAttribute::SetAttribute( const char* v )\n{\n    _value.SetStr( v );\n}\n\n\nvoid XMLAttribute::SetAttribute( int v )\n{\n    char buf[BUF_SIZE];\n    XMLUtil::ToStr( v, buf, BUF_SIZE );\n    _value.SetStr( buf );\n}\n\n\nvoid XMLAttribute::SetAttribute( unsigned v )\n{\n    char buf[BUF_SIZE];\n    XMLUtil::ToStr( v, buf, BUF_SIZE );\n    _value.SetStr( buf );\n}\n\n\nvoid XMLAttribute::SetAttribute(int64_t v)\n{\n\tchar buf[BUF_SIZE];\n\tXMLUtil::ToStr(v, buf, BUF_SIZE);\n\t_value.SetStr(buf);\n}\n\nvoid XMLAttribute::SetAttribute(uint64_t v)\n{\n    char buf[BUF_SIZE];\n    XMLUtil::ToStr(v, buf, BUF_SIZE);\n    _value.SetStr(buf);\n}\n\n\nvoid XMLAttribute::SetAttribute( bool v )\n{\n    char buf[BUF_SIZE];\n    XMLUtil::ToStr( v, buf, BUF_SIZE );\n    _value.SetStr( buf );\n}\n\nvoid XMLAttribute::SetAttribute( double v )\n{\n    char buf[BUF_SIZE];\n    XMLUtil::ToStr( v, buf, BUF_SIZE );\n    _value.SetStr( buf );\n}\n\nvoid XMLAttribute::SetAttribute( float v )\n{\n    char buf[BUF_SIZE];\n    XMLUtil::ToStr( v, buf, BUF_SIZE );\n    _value.SetStr( buf );\n}\n\n\n// --------- XMLElement ---------- //\nXMLElement::XMLElement( XMLDocument* doc ) : XMLNode( doc ),\n    _closingType( OPEN ),\n    _rootAttribute( 0 )\n{\n}\n\n\nXMLElement::~XMLElement()\n{\n    while( _rootAttribute ) {\n        XMLAttribute* next = _rootAttribute->_next;\n        DeleteAttribute( _rootAttribute );\n        _rootAttribute = next;\n    }\n}\n\n\nconst XMLAttribute* XMLElement::FindAttribute( const char* name ) const\n{\n    for( XMLAttribute* a = _rootAttribute; a; a = a->_next ) {\n        if ( XMLUtil::StringEqual( a->Name(), name ) ) {\n            return a;\n        }\n    }\n    return 0;\n}\n\n\nconst char* XMLElement::Attribute( const char* name, const char* value ) const\n{\n    const XMLAttribute* a = FindAttribute( name );\n    if ( !a ) {\n        return 0;\n    }\n    if ( !value || XMLUtil::StringEqual( a->Value(), value )) {\n        return a->Value();\n    }\n    return 0;\n}\n\nint XMLElement::IntAttribute(const char* name, int defaultValue) const\n{\n\tint i = defaultValue;\n\tQueryIntAttribute(name, &i);\n\treturn i;\n}\n\nunsigned XMLElement::UnsignedAttribute(const char* name, unsigned defaultValue) const\n{\n\tunsigned i = defaultValue;\n\tQueryUnsignedAttribute(name, &i);\n\treturn i;\n}\n\nint64_t XMLElement::Int64Attribute(const char* name, int64_t defaultValue) const\n{\n\tint64_t i = defaultValue;\n\tQueryInt64Attribute(name, &i);\n\treturn i;\n}\n\nuint64_t XMLElement::Unsigned64Attribute(const char* name, uint64_t defaultValue) const\n{\n\tuint64_t i = defaultValue;\n\tQueryUnsigned64Attribute(name, &i);\n\treturn i;\n}\n\nbool XMLElement::BoolAttribute(const char* name, bool defaultValue) const\n{\n\tbool b = defaultValue;\n\tQueryBoolAttribute(name, &b);\n\treturn b;\n}\n\ndouble XMLElement::DoubleAttribute(const char* name, double defaultValue) const\n{\n\tdouble d = defaultValue;\n\tQueryDoubleAttribute(name, &d);\n\treturn d;\n}\n\nfloat XMLElement::FloatAttribute(const char* name, float defaultValue) const\n{\n\tfloat f = defaultValue;\n\tQueryFloatAttribute(name, &f);\n\treturn f;\n}\n\nconst char* XMLElement::GetText() const\n{\n    if ( FirstChild() && FirstChild()->ToText() ) {\n        return FirstChild()->Value();\n    }\n    return 0;\n}\n\n\nvoid\tXMLElement::SetText( const char* inText )\n{\n\tif ( FirstChild() && FirstChild()->ToText() )\n\t\tFirstChild()->SetValue( inText );\n\telse {\n\t\tXMLText*\ttheText = GetDocument()->NewText( inText );\n\t\tInsertFirstChild( theText );\n\t}\n}\n\n\nvoid XMLElement::SetText( int v )\n{\n    char buf[BUF_SIZE];\n    XMLUtil::ToStr( v, buf, BUF_SIZE );\n    SetText( buf );\n}\n\n\nvoid XMLElement::SetText( unsigned v )\n{\n    char buf[BUF_SIZE];\n    XMLUtil::ToStr( v, buf, BUF_SIZE );\n    SetText( buf );\n}\n\n\nvoid XMLElement::SetText(int64_t v)\n{\n\tchar buf[BUF_SIZE];\n\tXMLUtil::ToStr(v, buf, BUF_SIZE);\n\tSetText(buf);\n}\n\nvoid XMLElement::SetText(uint64_t v) {\n    char buf[BUF_SIZE];\n    XMLUtil::ToStr(v, buf, BUF_SIZE);\n    SetText(buf);\n}\n\n\nvoid XMLElement::SetText( bool v )\n{\n    char buf[BUF_SIZE];\n    XMLUtil::ToStr( v, buf, BUF_SIZE );\n    SetText( buf );\n}\n\n\nvoid XMLElement::SetText( float v )\n{\n    char buf[BUF_SIZE];\n    XMLUtil::ToStr( v, buf, BUF_SIZE );\n    SetText( buf );\n}\n\n\nvoid XMLElement::SetText( double v )\n{\n    char buf[BUF_SIZE];\n    XMLUtil::ToStr( v, buf, BUF_SIZE );\n    SetText( buf );\n}\n\n\nXMLError XMLElement::QueryIntText( int* ival ) const\n{\n    if ( FirstChild() && FirstChild()->ToText() ) {\n        const char* t = FirstChild()->Value();\n        if ( XMLUtil::ToInt( t, ival ) ) {\n            return XML_SUCCESS;\n        }\n        return XML_CAN_NOT_CONVERT_TEXT;\n    }\n    return XML_NO_TEXT_NODE;\n}\n\n\nXMLError XMLElement::QueryUnsignedText( unsigned* uval ) const\n{\n    if ( FirstChild() && FirstChild()->ToText() ) {\n        const char* t = FirstChild()->Value();\n        if ( XMLUtil::ToUnsigned( t, uval ) ) {\n            return XML_SUCCESS;\n        }\n        return XML_CAN_NOT_CONVERT_TEXT;\n    }\n    return XML_NO_TEXT_NODE;\n}\n\n\nXMLError XMLElement::QueryInt64Text(int64_t* ival) const\n{\n\tif (FirstChild() && FirstChild()->ToText()) {\n\t\tconst char* t = FirstChild()->Value();\n\t\tif (XMLUtil::ToInt64(t, ival)) {\n\t\t\treturn XML_SUCCESS;\n\t\t}\n\t\treturn XML_CAN_NOT_CONVERT_TEXT;\n\t}\n\treturn XML_NO_TEXT_NODE;\n}\n\n\nXMLError XMLElement::QueryUnsigned64Text(uint64_t* ival) const\n{\n    if(FirstChild() && FirstChild()->ToText()) {\n        const char* t = FirstChild()->Value();\n        if(XMLUtil::ToUnsigned64(t, ival)) {\n            return XML_SUCCESS;\n        }\n        return XML_CAN_NOT_CONVERT_TEXT;\n    }\n    return XML_NO_TEXT_NODE;\n}\n\n\nXMLError XMLElement::QueryBoolText( bool* bval ) const\n{\n    if ( FirstChild() && FirstChild()->ToText() ) {\n        const char* t = FirstChild()->Value();\n        if ( XMLUtil::ToBool( t, bval ) ) {\n            return XML_SUCCESS;\n        }\n        return XML_CAN_NOT_CONVERT_TEXT;\n    }\n    return XML_NO_TEXT_NODE;\n}\n\n\nXMLError XMLElement::QueryDoubleText( double* dval ) const\n{\n    if ( FirstChild() && FirstChild()->ToText() ) {\n        const char* t = FirstChild()->Value();\n        if ( XMLUtil::ToDouble( t, dval ) ) {\n            return XML_SUCCESS;\n        }\n        return XML_CAN_NOT_CONVERT_TEXT;\n    }\n    return XML_NO_TEXT_NODE;\n}\n\n\nXMLError XMLElement::QueryFloatText( float* fval ) const\n{\n    if ( FirstChild() && FirstChild()->ToText() ) {\n        const char* t = FirstChild()->Value();\n        if ( XMLUtil::ToFloat( t, fval ) ) {\n            return XML_SUCCESS;\n        }\n        return XML_CAN_NOT_CONVERT_TEXT;\n    }\n    return XML_NO_TEXT_NODE;\n}\n\nint XMLElement::IntText(int defaultValue) const\n{\n\tint i = defaultValue;\n\tQueryIntText(&i);\n\treturn i;\n}\n\nunsigned XMLElement::UnsignedText(unsigned defaultValue) const\n{\n\tunsigned i = defaultValue;\n\tQueryUnsignedText(&i);\n\treturn i;\n}\n\nint64_t XMLElement::Int64Text(int64_t defaultValue) const\n{\n\tint64_t i = defaultValue;\n\tQueryInt64Text(&i);\n\treturn i;\n}\n\nuint64_t XMLElement::Unsigned64Text(uint64_t defaultValue) const\n{\n\tuint64_t i = defaultValue;\n\tQueryUnsigned64Text(&i);\n\treturn i;\n}\n\nbool XMLElement::BoolText(bool defaultValue) const\n{\n\tbool b = defaultValue;\n\tQueryBoolText(&b);\n\treturn b;\n}\n\ndouble XMLElement::DoubleText(double defaultValue) const\n{\n\tdouble d = defaultValue;\n\tQueryDoubleText(&d);\n\treturn d;\n}\n\nfloat XMLElement::FloatText(float defaultValue) const\n{\n\tfloat f = defaultValue;\n\tQueryFloatText(&f);\n\treturn f;\n}\n\n\nXMLAttribute* XMLElement::FindOrCreateAttribute( const char* name )\n{\n    XMLAttribute* last = 0;\n    XMLAttribute* attrib = 0;\n    for( attrib = _rootAttribute;\n            attrib;\n            last = attrib, attrib = attrib->_next ) {\n        if ( XMLUtil::StringEqual( attrib->Name(), name ) ) {\n            break;\n        }\n    }\n    if ( !attrib ) {\n        attrib = CreateAttribute();\n        TIXMLASSERT( attrib );\n        if ( last ) {\n            TIXMLASSERT( last->_next == 0 );\n            last->_next = attrib;\n        }\n        else {\n            TIXMLASSERT( _rootAttribute == 0 );\n            _rootAttribute = attrib;\n        }\n        attrib->SetName( name );\n    }\n    return attrib;\n}\n\n\nvoid XMLElement::DeleteAttribute( const char* name )\n{\n    XMLAttribute* prev = 0;\n    for( XMLAttribute* a=_rootAttribute; a; a=a->_next ) {\n        if ( XMLUtil::StringEqual( name, a->Name() ) ) {\n            if ( prev ) {\n                prev->_next = a->_next;\n            }\n            else {\n                _rootAttribute = a->_next;\n            }\n            DeleteAttribute( a );\n            break;\n        }\n        prev = a;\n    }\n}\n\n\nchar* XMLElement::ParseAttributes( char* p, int* curLineNumPtr )\n{\n    XMLAttribute* prevAttribute = 0;\n\n    // Read the attributes.\n    while( p ) {\n        p = XMLUtil::SkipWhiteSpace( p, curLineNumPtr );\n        if ( !(*p) ) {\n            _document->SetError( XML_ERROR_PARSING_ELEMENT, _parseLineNum, \"XMLElement name=%s\", Name() );\n            return 0;\n        }\n\n        // attribute.\n        if (XMLUtil::IsNameStartChar( (unsigned char) *p ) ) {\n            XMLAttribute* attrib = CreateAttribute();\n            TIXMLASSERT( attrib );\n            attrib->_parseLineNum = _document->_parseCurLineNum;\n\n            const int attrLineNum = attrib->_parseLineNum;\n\n            p = attrib->ParseDeep( p, _document->ProcessEntities(), curLineNumPtr );\n            if ( !p || Attribute( attrib->Name() ) ) {\n                DeleteAttribute( attrib );\n                _document->SetError( XML_ERROR_PARSING_ATTRIBUTE, attrLineNum, \"XMLElement name=%s\", Name() );\n                return 0;\n            }\n            // There is a minor bug here: if the attribute in the source xml\n            // document is duplicated, it will not be detected and the\n            // attribute will be doubly added. However, tracking the 'prevAttribute'\n            // avoids re-scanning the attribute list. Preferring performance for\n            // now, may reconsider in the future.\n            if ( prevAttribute ) {\n                TIXMLASSERT( prevAttribute->_next == 0 );\n                prevAttribute->_next = attrib;\n            }\n            else {\n                TIXMLASSERT( _rootAttribute == 0 );\n                _rootAttribute = attrib;\n            }\n            prevAttribute = attrib;\n        }\n        // end of the tag\n        else if ( *p == '>' ) {\n            ++p;\n            break;\n        }\n        // end of the tag\n        else if ( *p == '/' && *(p+1) == '>' ) {\n            _closingType = CLOSED;\n            return p+2;\t// done; sealed element.\n        }\n        else {\n            _document->SetError( XML_ERROR_PARSING_ELEMENT, _parseLineNum, 0 );\n            return 0;\n        }\n    }\n    return p;\n}\n\nvoid XMLElement::DeleteAttribute( XMLAttribute* attribute )\n{\n    if ( attribute == 0 ) {\n        return;\n    }\n    MemPool* pool = attribute->_memPool;\n    attribute->~XMLAttribute();\n    pool->Free( attribute );\n}\n\nXMLAttribute* XMLElement::CreateAttribute()\n{\n    TIXMLASSERT( sizeof( XMLAttribute ) == _document->_attributePool.ItemSize() );\n    XMLAttribute* attrib = new (_document->_attributePool.Alloc() ) XMLAttribute();\n    TIXMLASSERT( attrib );\n    attrib->_memPool = &_document->_attributePool;\n    attrib->_memPool->SetTracked();\n    return attrib;\n}\n\n\nXMLElement* XMLElement::InsertNewChildElement(const char* name)\n{\n    XMLElement* node = _document->NewElement(name);\n    return InsertEndChild(node) ? node : 0;\n}\n\nXMLComment* XMLElement::InsertNewComment(const char* comment)\n{\n    XMLComment* node = _document->NewComment(comment);\n    return InsertEndChild(node) ? node : 0;\n}\n\nXMLText* XMLElement::InsertNewText(const char* text)\n{\n    XMLText* node = _document->NewText(text);\n    return InsertEndChild(node) ? node : 0;\n}\n\nXMLDeclaration* XMLElement::InsertNewDeclaration(const char* text)\n{\n    XMLDeclaration* node = _document->NewDeclaration(text);\n    return InsertEndChild(node) ? node : 0;\n}\n\nXMLUnknown* XMLElement::InsertNewUnknown(const char* text)\n{\n    XMLUnknown* node = _document->NewUnknown(text);\n    return InsertEndChild(node) ? node : 0;\n}\n\n\n\n//\n//\t<ele></ele>\n//\t<ele>foo<b>bar</b></ele>\n//\nchar* XMLElement::ParseDeep( char* p, StrPair* parentEndTag, int* curLineNumPtr )\n{\n    // Read the element name.\n    p = XMLUtil::SkipWhiteSpace( p, curLineNumPtr );\n\n    // The closing element is the </element> form. It is\n    // parsed just like a regular element then deleted from\n    // the DOM.\n    if ( *p == '/' ) {\n        _closingType = CLOSING;\n        ++p;\n    }\n\n    p = _value.ParseName( p );\n    if ( _value.Empty() ) {\n        return 0;\n    }\n\n    p = ParseAttributes( p, curLineNumPtr );\n    if ( !p || !*p || _closingType != OPEN ) {\n        return p;\n    }\n\n    p = XMLNode::ParseDeep( p, parentEndTag, curLineNumPtr );\n    return p;\n}\n\n\n\nXMLNode* XMLElement::ShallowClone( XMLDocument* doc ) const\n{\n    if ( !doc ) {\n        doc = _document;\n    }\n    XMLElement* element = doc->NewElement( Value() );\t\t\t\t\t// fixme: this will always allocate memory. Intern?\n    for( const XMLAttribute* a=FirstAttribute(); a; a=a->Next() ) {\n        element->SetAttribute( a->Name(), a->Value() );\t\t\t\t\t// fixme: this will always allocate memory. Intern?\n    }\n    return element;\n}\n\n\nbool XMLElement::ShallowEqual( const XMLNode* compare ) const\n{\n    TIXMLASSERT( compare );\n    const XMLElement* other = compare->ToElement();\n    if ( other && XMLUtil::StringEqual( other->Name(), Name() )) {\n\n        const XMLAttribute* a=FirstAttribute();\n        const XMLAttribute* b=other->FirstAttribute();\n\n        while ( a && b ) {\n            if ( !XMLUtil::StringEqual( a->Value(), b->Value() ) ) {\n                return false;\n            }\n            a = a->Next();\n            b = b->Next();\n        }\n        if ( a || b ) {\n            // different count\n            return false;\n        }\n        return true;\n    }\n    return false;\n}\n\n\nbool XMLElement::Accept( XMLVisitor* visitor ) const\n{\n    TIXMLASSERT( visitor );\n    if ( visitor->VisitEnter( *this, _rootAttribute ) ) {\n        for ( const XMLNode* node=FirstChild(); node; node=node->NextSibling() ) {\n            if ( !node->Accept( visitor ) ) {\n                break;\n            }\n        }\n    }\n    return visitor->VisitExit( *this );\n}\n\n\n// --------- XMLDocument ----------- //\n\n// Warning: List must match 'enum XMLError'\nconst char* XMLDocument::_errorNames[XML_ERROR_COUNT] = {\n    \"XML_SUCCESS\",\n    \"XML_NO_ATTRIBUTE\",\n    \"XML_WRONG_ATTRIBUTE_TYPE\",\n    \"XML_ERROR_FILE_NOT_FOUND\",\n    \"XML_ERROR_FILE_COULD_NOT_BE_OPENED\",\n    \"XML_ERROR_FILE_READ_ERROR\",\n    \"XML_ERROR_PARSING_ELEMENT\",\n    \"XML_ERROR_PARSING_ATTRIBUTE\",\n    \"XML_ERROR_PARSING_TEXT\",\n    \"XML_ERROR_PARSING_CDATA\",\n    \"XML_ERROR_PARSING_COMMENT\",\n    \"XML_ERROR_PARSING_DECLARATION\",\n    \"XML_ERROR_PARSING_UNKNOWN\",\n    \"XML_ERROR_EMPTY_DOCUMENT\",\n    \"XML_ERROR_MISMATCHED_ELEMENT\",\n    \"XML_ERROR_PARSING\",\n    \"XML_CAN_NOT_CONVERT_TEXT\",\n    \"XML_NO_TEXT_NODE\",\n\t\"XML_ELEMENT_DEPTH_EXCEEDED\"\n};\n\n\nXMLDocument::XMLDocument( bool processEntities, Whitespace whitespaceMode ) :\n    XMLNode( 0 ),\n    _writeBOM( false ),\n    _processEntities( processEntities ),\n    _errorID(XML_SUCCESS),\n    _whitespaceMode( whitespaceMode ),\n    _errorStr(),\n    _errorLineNum( 0 ),\n    _charBuffer( 0 ),\n    _parseCurLineNum( 0 ),\n\t_parsingDepth(0),\n    _unlinked(),\n    _elementPool(),\n    _attributePool(),\n    _textPool(),\n    _commentPool()\n{\n    // avoid VC++ C4355 warning about 'this' in initializer list (C4355 is off by default in VS2012+)\n    _document = this;\n}\n\n\nXMLDocument::~XMLDocument()\n{\n    Clear();\n}\n\n\nvoid XMLDocument::MarkInUse(const XMLNode* const node)\n{\n\tTIXMLASSERT(node);\n\tTIXMLASSERT(node->_parent == 0);\n\n\tfor (int i = 0; i < _unlinked.Size(); ++i) {\n\t\tif (node == _unlinked[i]) {\n\t\t\t_unlinked.SwapRemove(i);\n\t\t\tbreak;\n\t\t}\n\t}\n}\n\nvoid XMLDocument::Clear()\n{\n    DeleteChildren();\n\twhile( _unlinked.Size()) {\n\t\tDeleteNode(_unlinked[0]);\t// Will remove from _unlinked as part of delete.\n\t}\n\n#ifdef TINYXML2_DEBUG\n    const bool hadError = Error();\n#endif\n    ClearError();\n\n    delete [] _charBuffer;\n    _charBuffer = 0;\n\t_parsingDepth = 0;\n\n#if 0\n    _textPool.Trace( \"text\" );\n    _elementPool.Trace( \"element\" );\n    _commentPool.Trace( \"comment\" );\n    _attributePool.Trace( \"attribute\" );\n#endif\n\n#ifdef TINYXML2_DEBUG\n    if ( !hadError ) {\n        TIXMLASSERT( _elementPool.CurrentAllocs()   == _elementPool.Untracked() );\n        TIXMLASSERT( _attributePool.CurrentAllocs() == _attributePool.Untracked() );\n        TIXMLASSERT( _textPool.CurrentAllocs()      == _textPool.Untracked() );\n        TIXMLASSERT( _commentPool.CurrentAllocs()   == _commentPool.Untracked() );\n    }\n#endif\n}\n\n\nvoid XMLDocument::DeepCopy(XMLDocument* target) const\n{\n\tTIXMLASSERT(target);\n    if (target == this) {\n        return; // technically success - a no-op.\n    }\n\n\ttarget->Clear();\n\tfor (const XMLNode* node = this->FirstChild(); node; node = node->NextSibling()) {\n\t\ttarget->InsertEndChild(node->DeepClone(target));\n\t}\n}\n\nXMLElement* XMLDocument::NewElement( const char* name )\n{\n    XMLElement* ele = CreateUnlinkedNode<XMLElement>( _elementPool );\n    ele->SetName( name );\n    return ele;\n}\n\n\nXMLComment* XMLDocument::NewComment( const char* str )\n{\n    XMLComment* comment = CreateUnlinkedNode<XMLComment>( _commentPool );\n    comment->SetValue( str );\n    return comment;\n}\n\n\nXMLText* XMLDocument::NewText( const char* str )\n{\n    XMLText* text = CreateUnlinkedNode<XMLText>( _textPool );\n    text->SetValue( str );\n    return text;\n}\n\n\nXMLDeclaration* XMLDocument::NewDeclaration( const char* str )\n{\n    XMLDeclaration* dec = CreateUnlinkedNode<XMLDeclaration>( _commentPool );\n    dec->SetValue( str ? str : \"xml version=\\\"1.0\\\" encoding=\\\"UTF-8\\\"\" );\n    return dec;\n}\n\n\nXMLUnknown* XMLDocument::NewUnknown( const char* str )\n{\n    XMLUnknown* unk = CreateUnlinkedNode<XMLUnknown>( _commentPool );\n    unk->SetValue( str );\n    return unk;\n}\n\nstatic FILE* callfopen( const char* filepath, const char* mode )\n{\n    TIXMLASSERT( filepath );\n    TIXMLASSERT( mode );\n#if defined(_MSC_VER) && (_MSC_VER >= 1400 ) && (!defined WINCE)\n    FILE* fp = 0;\n    const errno_t err = fopen_s( &fp, filepath, mode );\n    if ( err ) {\n        return 0;\n    }\n#else\n    FILE* fp = fopen( filepath, mode );\n#endif\n    return fp;\n}\n\nvoid XMLDocument::DeleteNode( XMLNode* node )\t{\n    TIXMLASSERT( node );\n    TIXMLASSERT(node->_document == this );\n    if (node->_parent) {\n        node->_parent->DeleteChild( node );\n    }\n    else {\n        // Isn't in the tree.\n        // Use the parent delete.\n        // Also, we need to mark it tracked: we 'know'\n        // it was never used.\n        node->_memPool->SetTracked();\n        // Call the static XMLNode version:\n        XMLNode::DeleteNode(node);\n    }\n}\n\n\nXMLError XMLDocument::LoadFile( const char* filename )\n{\n    if ( !filename ) {\n        TIXMLASSERT( false );\n        SetError( XML_ERROR_FILE_COULD_NOT_BE_OPENED, 0, \"filename=<null>\" );\n        return _errorID;\n    }\n\n    Clear();\n    FILE* fp = callfopen( filename, \"rb\" );\n    if ( !fp ) {\n        SetError( XML_ERROR_FILE_NOT_FOUND, 0, \"filename=%s\", filename );\n        return _errorID;\n    }\n    LoadFile( fp );\n    fclose( fp );\n    return _errorID;\n}\n\nXMLError XMLDocument::LoadFile( FILE* fp )\n{\n    Clear();\n\n    TIXML_FSEEK( fp, 0, SEEK_SET );\n    if ( fgetc( fp ) == EOF && ferror( fp ) != 0 ) {\n        SetError( XML_ERROR_FILE_READ_ERROR, 0, 0 );\n        return _errorID;\n    }\n\n    TIXML_FSEEK( fp, 0, SEEK_END );\n\n    unsigned long long filelength;\n    {\n        const long long fileLengthSigned = TIXML_FTELL( fp );\n        TIXML_FSEEK( fp, 0, SEEK_SET );\n        if ( fileLengthSigned == -1L ) {\n            SetError( XML_ERROR_FILE_READ_ERROR, 0, 0 );\n            return _errorID;\n        }\n        TIXMLASSERT( fileLengthSigned >= 0 );\n        filelength = static_cast<unsigned long long>(fileLengthSigned);\n    }\n\n    const size_t maxSizeT = static_cast<size_t>(-1);\n    // We'll do the comparison as an unsigned long long, because that's guaranteed to be at\n    // least 8 bytes, even on a 32-bit platform.\n    if ( filelength >= static_cast<unsigned long long>(maxSizeT) ) {\n        // Cannot handle files which won't fit in buffer together with null terminator\n        SetError( XML_ERROR_FILE_READ_ERROR, 0, 0 );\n        return _errorID;\n    }\n\n    if ( filelength == 0 ) {\n        SetError( XML_ERROR_EMPTY_DOCUMENT, 0, 0 );\n        return _errorID;\n    }\n\n    const size_t size = static_cast<size_t>(filelength);\n    TIXMLASSERT( _charBuffer == 0 );\n    _charBuffer = new char[size+1];\n    const size_t read = fread( _charBuffer, 1, size, fp );\n    if ( read != size ) {\n        SetError( XML_ERROR_FILE_READ_ERROR, 0, 0 );\n        return _errorID;\n    }\n\n    _charBuffer[size] = 0;\n\n    Parse();\n    return _errorID;\n}\n\n\nXMLError XMLDocument::SaveFile( const char* filename, bool compact )\n{\n    if ( !filename ) {\n        TIXMLASSERT( false );\n        SetError( XML_ERROR_FILE_COULD_NOT_BE_OPENED, 0, \"filename=<null>\" );\n        return _errorID;\n    }\n\n    FILE* fp = callfopen( filename, \"w\" );\n    if ( !fp ) {\n        SetError( XML_ERROR_FILE_COULD_NOT_BE_OPENED, 0, \"filename=%s\", filename );\n        return _errorID;\n    }\n    SaveFile(fp, compact);\n    fclose( fp );\n    return _errorID;\n}\n\n\nXMLError XMLDocument::SaveFile( FILE* fp, bool compact )\n{\n    // Clear any error from the last save, otherwise it will get reported\n    // for *this* call.\n    ClearError();\n    XMLPrinter stream( fp, compact );\n    Print( &stream );\n    return _errorID;\n}\n\n\nXMLError XMLDocument::Parse( const char* p, size_t len )\n{\n    Clear();\n\n    if ( len == 0 || !p || !*p ) {\n        SetError( XML_ERROR_EMPTY_DOCUMENT, 0, 0 );\n        return _errorID;\n    }\n    if ( len == static_cast<size_t>(-1) ) {\n        len = strlen( p );\n    }\n    TIXMLASSERT( _charBuffer == 0 );\n    _charBuffer = new char[ len+1 ];\n    memcpy( _charBuffer, p, len );\n    _charBuffer[len] = 0;\n\n    Parse();\n    if ( Error() ) {\n        // clean up now essentially dangling memory.\n        // and the parse fail can put objects in the\n        // pools that are dead and inaccessible.\n        DeleteChildren();\n        _elementPool.Clear();\n        _attributePool.Clear();\n        _textPool.Clear();\n        _commentPool.Clear();\n    }\n    return _errorID;\n}\n\n\nvoid XMLDocument::Print( XMLPrinter* streamer ) const\n{\n    if ( streamer ) {\n        Accept( streamer );\n    }\n    else {\n        XMLPrinter stdoutStreamer( stdout );\n        Accept( &stdoutStreamer );\n    }\n}\n\n\nvoid XMLDocument::SetError( XMLError error, int lineNum, const char* format, ... )\n{\n    TIXMLASSERT( error >= 0 && error < XML_ERROR_COUNT );\n    _errorID = error;\n    _errorLineNum = lineNum;\n\t_errorStr.Reset();\n\n    const size_t BUFFER_SIZE = 1000;\n    char* buffer = new char[BUFFER_SIZE];\n\n    TIXMLASSERT(sizeof(error) <= sizeof(int));\n    TIXML_SNPRINTF(buffer, BUFFER_SIZE, \"Error=%s ErrorID=%d (0x%x) Line number=%d\", ErrorIDToName(error), int(error), int(error), lineNum);\n\n\tif (format) {\n\t\tsize_t len = strlen(buffer);\n\t\tTIXML_SNPRINTF(buffer + len, BUFFER_SIZE - len, \": \");\n\t\tlen = strlen(buffer);\n\n\t\tva_list va;\n\t\tva_start(va, format);\n\t\tTIXML_VSNPRINTF(buffer + len, BUFFER_SIZE - len, format, va);\n\t\tva_end(va);\n\t}\n\t_errorStr.SetStr(buffer);\n\tdelete[] buffer;\n}\n\n\n/*static*/ const char* XMLDocument::ErrorIDToName(XMLError errorID)\n{\n\tTIXMLASSERT( errorID >= 0 && errorID < XML_ERROR_COUNT );\n    const char* errorName = _errorNames[errorID];\n    TIXMLASSERT( errorName && errorName[0] );\n    return errorName;\n}\n\nconst char* XMLDocument::ErrorStr() const\n{\n\treturn _errorStr.Empty() ? \"\" : _errorStr.GetStr();\n}\n\n\nvoid XMLDocument::PrintError() const\n{\n    printf(\"%s\\n\", ErrorStr());\n}\n\nconst char* XMLDocument::ErrorName() const\n{\n    return ErrorIDToName(_errorID);\n}\n\nvoid XMLDocument::Parse()\n{\n    TIXMLASSERT( NoChildren() ); // Clear() must have been called previously\n    TIXMLASSERT( _charBuffer );\n    _parseCurLineNum = 1;\n    _parseLineNum = 1;\n    char* p = _charBuffer;\n    p = XMLUtil::SkipWhiteSpace( p, &_parseCurLineNum );\n    p = const_cast<char*>( XMLUtil::ReadBOM( p, &_writeBOM ) );\n    if ( !*p ) {\n        SetError( XML_ERROR_EMPTY_DOCUMENT, 0, 0 );\n        return;\n    }\n    ParseDeep(p, 0, &_parseCurLineNum );\n}\n\nvoid XMLDocument::PushDepth()\n{\n\t_parsingDepth++;\n\tif (_parsingDepth == TINYXML2_MAX_ELEMENT_DEPTH) {\n\t\tSetError(XML_ELEMENT_DEPTH_EXCEEDED, _parseCurLineNum, \"Element nesting is too deep.\" );\n\t}\n}\n\nvoid XMLDocument::PopDepth()\n{\n\tTIXMLASSERT(_parsingDepth > 0);\n\t--_parsingDepth;\n}\n\nXMLPrinter::XMLPrinter( FILE* file, bool compact, int depth ) :\n    _elementJustOpened( false ),\n    _stack(),\n    _firstElement( true ),\n    _fp( file ),\n    _depth( depth ),\n    _textDepth( -1 ),\n    _processEntities( true ),\n    _compactMode( compact ),\n    _buffer()\n{\n    for( int i=0; i<ENTITY_RANGE; ++i ) {\n        _entityFlag[i] = false;\n        _restrictedEntityFlag[i] = false;\n    }\n    for( int i=0; i<NUM_ENTITIES; ++i ) {\n        const char entityValue = entities[i].value;\n        const unsigned char flagIndex = static_cast<unsigned char>(entityValue);\n        TIXMLASSERT( flagIndex < ENTITY_RANGE );\n        _entityFlag[flagIndex] = true;\n    }\n    _restrictedEntityFlag[static_cast<unsigned char>('&')] = true;\n    _restrictedEntityFlag[static_cast<unsigned char>('<')] = true;\n    _restrictedEntityFlag[static_cast<unsigned char>('>')] = true;\t// not required, but consistency is nice\n    _buffer.Push( 0 );\n}\n\n\nvoid XMLPrinter::Print( const char* format, ... )\n{\n    va_list     va;\n    va_start( va, format );\n\n    if ( _fp ) {\n        vfprintf( _fp, format, va );\n    }\n    else {\n        const int len = TIXML_VSCPRINTF( format, va );\n        // Close out and re-start the va-args\n        va_end( va );\n        TIXMLASSERT( len >= 0 );\n        va_start( va, format );\n        TIXMLASSERT( _buffer.Size() > 0 && _buffer[_buffer.Size() - 1] == 0 );\n        char* p = _buffer.PushArr( len ) - 1;\t// back up over the null terminator.\n\t\tTIXML_VSNPRINTF( p, len+1, format, va );\n    }\n    va_end( va );\n}\n\n\nvoid XMLPrinter::Write( const char* data, size_t size )\n{\n    if ( _fp ) {\n        fwrite ( data , sizeof(char), size, _fp);\n    }\n    else {\n        char* p = _buffer.PushArr( static_cast<int>(size) ) - 1;   // back up over the null terminator.\n        memcpy( p, data, size );\n        p[size] = 0;\n    }\n}\n\n\nvoid XMLPrinter::Putc( char ch )\n{\n    if ( _fp ) {\n        fputc ( ch, _fp);\n    }\n    else {\n        char* p = _buffer.PushArr( sizeof(char) ) - 1;   // back up over the null terminator.\n        p[0] = ch;\n        p[1] = 0;\n    }\n}\n\n\nvoid XMLPrinter::PrintSpace( int depth )\n{\n    for( int i=0; i<depth; ++i ) {\n        Write( \"    \" );\n    }\n}\n\n\nvoid XMLPrinter::PrintString( const char* p, bool restricted )\n{\n    // Look for runs of bytes between entities to print.\n    const char* q = p;\n\n    if ( _processEntities ) {\n        const bool* flag = restricted ? _restrictedEntityFlag : _entityFlag;\n        while ( *q ) {\n            TIXMLASSERT( p <= q );\n            // Remember, char is sometimes signed. (How many times has that bitten me?)\n            if ( *q > 0 && *q < ENTITY_RANGE ) {\n                // Check for entities. If one is found, flush\n                // the stream up until the entity, write the\n                // entity, and keep looking.\n                if ( flag[static_cast<unsigned char>(*q)] ) {\n                    while ( p < q ) {\n                        const size_t delta = q - p;\n                        const int toPrint = ( INT_MAX < delta ) ? INT_MAX : static_cast<int>(delta);\n                        Write( p, toPrint );\n                        p += toPrint;\n                    }\n                    bool entityPatternPrinted = false;\n                    for( int i=0; i<NUM_ENTITIES; ++i ) {\n                        if ( entities[i].value == *q ) {\n                            Putc( '&' );\n                            Write( entities[i].pattern, entities[i].length );\n                            Putc( ';' );\n                            entityPatternPrinted = true;\n                            break;\n                        }\n                    }\n                    if ( !entityPatternPrinted ) {\n                        // TIXMLASSERT( entityPatternPrinted ) causes gcc -Wunused-but-set-variable in release\n                        TIXMLASSERT( false );\n                    }\n                    ++p;\n                }\n            }\n            ++q;\n            TIXMLASSERT( p <= q );\n        }\n        // Flush the remaining string. This will be the entire\n        // string if an entity wasn't found.\n        if ( p < q ) {\n            const size_t delta = q - p;\n            const int toPrint = ( INT_MAX < delta ) ? INT_MAX : static_cast<int>(delta);\n            Write( p, toPrint );\n        }\n    }\n    else {\n        Write( p );\n    }\n}\n\n\nvoid XMLPrinter::PushHeader( bool writeBOM, bool writeDec )\n{\n    if ( writeBOM ) {\n        static const unsigned char bom[] = { TIXML_UTF_LEAD_0, TIXML_UTF_LEAD_1, TIXML_UTF_LEAD_2, 0 };\n        Write( reinterpret_cast< const char* >( bom ) );\n    }\n    if ( writeDec ) {\n        PushDeclaration( \"xml version=\\\"1.0\\\"\" );\n    }\n}\n\nvoid XMLPrinter::PrepareForNewNode( bool compactMode )\n{\n    SealElementIfJustOpened();\n\n    if ( compactMode ) {\n        return;\n    }\n\n    if ( _firstElement ) {\n        PrintSpace (_depth);\n    } else if ( _textDepth < 0) {\n        Putc( '\\n' );\n        PrintSpace( _depth );\n    }\n\n    _firstElement = false;\n}\n\nvoid XMLPrinter::OpenElement( const char* name, bool compactMode )\n{\n    PrepareForNewNode( compactMode );\n    _stack.Push( name );\n\n    Write ( \"<\" );\n    Write ( name );\n\n    _elementJustOpened = true;\n    ++_depth;\n}\n\n\nvoid XMLPrinter::PushAttribute( const char* name, const char* value )\n{\n    TIXMLASSERT( _elementJustOpened );\n    Putc ( ' ' );\n    Write( name );\n    Write( \"=\\\"\" );\n    PrintString( value, false );\n    Putc ( '\\\"' );\n}\n\n\nvoid XMLPrinter::PushAttribute( const char* name, int v )\n{\n    char buf[BUF_SIZE];\n    XMLUtil::ToStr( v, buf, BUF_SIZE );\n    PushAttribute( name, buf );\n}\n\n\nvoid XMLPrinter::PushAttribute( const char* name, unsigned v )\n{\n    char buf[BUF_SIZE];\n    XMLUtil::ToStr( v, buf, BUF_SIZE );\n    PushAttribute( name, buf );\n}\n\n\nvoid XMLPrinter::PushAttribute(const char* name, int64_t v)\n{\n\tchar buf[BUF_SIZE];\n\tXMLUtil::ToStr(v, buf, BUF_SIZE);\n\tPushAttribute(name, buf);\n}\n\n\nvoid XMLPrinter::PushAttribute(const char* name, uint64_t v)\n{\n\tchar buf[BUF_SIZE];\n\tXMLUtil::ToStr(v, buf, BUF_SIZE);\n\tPushAttribute(name, buf);\n}\n\n\nvoid XMLPrinter::PushAttribute( const char* name, bool v )\n{\n    char buf[BUF_SIZE];\n    XMLUtil::ToStr( v, buf, BUF_SIZE );\n    PushAttribute( name, buf );\n}\n\n\nvoid XMLPrinter::PushAttribute( const char* name, double v )\n{\n    char buf[BUF_SIZE];\n    XMLUtil::ToStr( v, buf, BUF_SIZE );\n    PushAttribute( name, buf );\n}\n\n\nvoid XMLPrinter::CloseElement( bool compactMode )\n{\n    --_depth;\n    const char* name = _stack.Pop();\n\n    if ( _elementJustOpened ) {\n        Write( \"/>\" );\n    }\n    else {\n        if ( _textDepth < 0 && !compactMode) {\n            Putc( '\\n' );\n            PrintSpace( _depth );\n        }\n        Write ( \"</\" );\n        Write ( name );\n        Write ( \">\" );\n    }\n\n    if ( _textDepth == _depth ) {\n        _textDepth = -1;\n    }\n    if ( _depth == 0 && !compactMode) {\n        Putc( '\\n' );\n    }\n    _elementJustOpened = false;\n}\n\n\nvoid XMLPrinter::SealElementIfJustOpened()\n{\n    if ( !_elementJustOpened ) {\n        return;\n    }\n    _elementJustOpened = false;\n    Putc( '>' );\n}\n\n\nvoid XMLPrinter::PushText( const char* text, bool cdata )\n{\n    _textDepth = _depth-1;\n\n    SealElementIfJustOpened();\n    if ( cdata ) {\n        Write( \"<![CDATA[\" );\n        Write( text );\n        Write( \"]]>\" );\n    }\n    else {\n        PrintString( text, true );\n    }\n}\n\n\nvoid XMLPrinter::PushText( int64_t value )\n{\n    char buf[BUF_SIZE];\n    XMLUtil::ToStr( value, buf, BUF_SIZE );\n    PushText( buf, false );\n}\n\n\nvoid XMLPrinter::PushText( uint64_t value )\n{\n\tchar buf[BUF_SIZE];\n\tXMLUtil::ToStr(value, buf, BUF_SIZE);\n\tPushText(buf, false);\n}\n\n\nvoid XMLPrinter::PushText( int value )\n{\n    char buf[BUF_SIZE];\n    XMLUtil::ToStr( value, buf, BUF_SIZE );\n    PushText( buf, false );\n}\n\n\nvoid XMLPrinter::PushText( unsigned value )\n{\n    char buf[BUF_SIZE];\n    XMLUtil::ToStr( value, buf, BUF_SIZE );\n    PushText( buf, false );\n}\n\n\nvoid XMLPrinter::PushText( bool value )\n{\n    char buf[BUF_SIZE];\n    XMLUtil::ToStr( value, buf, BUF_SIZE );\n    PushText( buf, false );\n}\n\n\nvoid XMLPrinter::PushText( float value )\n{\n    char buf[BUF_SIZE];\n    XMLUtil::ToStr( value, buf, BUF_SIZE );\n    PushText( buf, false );\n}\n\n\nvoid XMLPrinter::PushText( double value )\n{\n    char buf[BUF_SIZE];\n    XMLUtil::ToStr( value, buf, BUF_SIZE );\n    PushText( buf, false );\n}\n\n\nvoid XMLPrinter::PushComment( const char* comment )\n{\n    PrepareForNewNode( _compactMode );\n\n    Write( \"<!--\" );\n    Write( comment );\n    Write( \"-->\" );\n}\n\n\nvoid XMLPrinter::PushDeclaration( const char* value )\n{\n    PrepareForNewNode( _compactMode );\n\n    Write( \"<?\" );\n    Write( value );\n    Write( \"?>\" );\n}\n\n\nvoid XMLPrinter::PushUnknown( const char* value )\n{\n    PrepareForNewNode( _compactMode );\n\n    Write( \"<!\" );\n    Write( value );\n    Putc( '>' );\n}\n\n\nbool XMLPrinter::VisitEnter( const XMLDocument& doc )\n{\n    _processEntities = doc.ProcessEntities();\n    if ( doc.HasBOM() ) {\n        PushHeader( true, false );\n    }\n    return true;\n}\n\n\nbool XMLPrinter::VisitEnter( const XMLElement& element, const XMLAttribute* attribute )\n{\n    const XMLElement* parentElem = 0;\n    if ( element.Parent() ) {\n        parentElem = element.Parent()->ToElement();\n    }\n    const bool compactMode = parentElem ? CompactMode( *parentElem ) : _compactMode;\n    OpenElement( element.Name(), compactMode );\n    while ( attribute ) {\n        PushAttribute( attribute->Name(), attribute->Value() );\n        attribute = attribute->Next();\n    }\n    return true;\n}\n\n\nbool XMLPrinter::VisitExit( const XMLElement& element )\n{\n    CloseElement( CompactMode(element) );\n    return true;\n}\n\n\nbool XMLPrinter::Visit( const XMLText& text )\n{\n    PushText( text.Value(), text.CData() );\n    return true;\n}\n\n\nbool XMLPrinter::Visit( const XMLComment& comment )\n{\n    PushComment( comment.Value() );\n    return true;\n}\n\nbool XMLPrinter::Visit( const XMLDeclaration& declaration )\n{\n    PushDeclaration( declaration.Value() );\n    return true;\n}\n\n\nbool XMLPrinter::Visit( const XMLUnknown& unknown )\n{\n    PushUnknown( unknown.Value() );\n    return true;\n}\n\n}   // namespace tinyxml2\n"
  },
  {
    "path": "TrafficMonitor/tinyxml2/tinyxml2.h",
    "content": "/*\nOriginal code by Lee Thomason (www.grinninglizard.com)\n\nThis software is provided 'as-is', without any express or implied\nwarranty. In no event will the authors be held liable for any\ndamages arising from the use of this software.\n\nPermission is granted to anyone to use this software for any\npurpose, including commercial applications, and to alter it and\nredistribute it freely, subject to the following restrictions:\n\n1. The origin of this software must not be misrepresented; you must\nnot claim that you wrote the original software. If you use this\nsoftware in a product, an acknowledgment in the product documentation\nwould be appreciated but is not required.\n\n2. Altered source versions must be plainly marked as such, and\nmust not be misrepresented as being the original software.\n\n3. This notice may not be removed or altered from any source\ndistribution.\n*/\n\n#ifndef TINYXML2_INCLUDED\n#define TINYXML2_INCLUDED\n\n#if defined(ANDROID_NDK) || defined(__BORLANDC__) || defined(__QNXNTO__)\n#   include <ctype.h>\n#   include <limits.h>\n#   include <stdio.h>\n#   include <stdlib.h>\n#   include <string.h>\n#\tif defined(__PS3__)\n#\t\tinclude <stddef.h>\n#\tendif\n#else\n#   include <cctype>\n#   include <climits>\n#   include <cstdio>\n#   include <cstdlib>\n#   include <cstring>\n#endif\n#include <stdint.h>\n\n/*\n   TODO: intern strings instead of allocation.\n*/\n/*\n\tgcc:\n        g++ -Wall -DTINYXML2_DEBUG tinyxml2.cpp xmltest.cpp -o gccxmltest.exe\n\n    Formatting, Artistic Style:\n        AStyle.exe --style=1tbs --indent-switches --break-closing-brackets --indent-preprocessor tinyxml2.cpp tinyxml2.h\n*/\n\n#if defined( _DEBUG ) || defined (__DEBUG__)\n#   ifndef TINYXML2_DEBUG\n#       define TINYXML2_DEBUG\n#   endif\n#endif\n\n#ifdef _MSC_VER\n#   pragma warning(push)\n#   pragma warning(disable: 4251)\n#endif\n\n#ifdef _WIN32\n#   ifdef TINYXML2_EXPORT\n#       define TINYXML2_LIB __declspec(dllexport)\n#   elif defined(TINYXML2_IMPORT)\n#       define TINYXML2_LIB __declspec(dllimport)\n#   else\n#       define TINYXML2_LIB\n#   endif\n#elif __GNUC__ >= 4\n#   define TINYXML2_LIB __attribute__((visibility(\"default\")))\n#else\n#   define TINYXML2_LIB\n#endif\n\n\n#if defined(TINYXML2_DEBUG)\n#   if defined(_MSC_VER)\n#       // \"(void)0,\" is for suppressing C4127 warning in \"assert(false)\", \"assert(true)\" and the like\n#       define TIXMLASSERT( x )           if ( !((void)0,(x))) { __debugbreak(); }\n#   elif defined (ANDROID_NDK)\n#       include <android/log.h>\n#       define TIXMLASSERT( x )           if ( !(x)) { __android_log_assert( \"assert\", \"grinliz\", \"ASSERT in '%s' at %d.\", __FILE__, __LINE__ ); }\n#   else\n#       include <assert.h>\n#       define TIXMLASSERT                assert\n#   endif\n#else\n#   define TIXMLASSERT( x )               {}\n#endif\n\n\n/* Versioning, past 1.0.14:\n\thttp://semver.org/\n*/\nstatic const int TIXML2_MAJOR_VERSION = 8;\nstatic const int TIXML2_MINOR_VERSION = 0;\nstatic const int TIXML2_PATCH_VERSION = 0;\n\n#define TINYXML2_MAJOR_VERSION 8\n#define TINYXML2_MINOR_VERSION 0\n#define TINYXML2_PATCH_VERSION 0\n\n// A fixed element depth limit is problematic. There needs to be a\n// limit to avoid a stack overflow. However, that limit varies per\n// system, and the capacity of the stack. On the other hand, it's a trivial\n// attack that can result from ill, malicious, or even correctly formed XML,\n// so there needs to be a limit in place.\nstatic const int TINYXML2_MAX_ELEMENT_DEPTH = 100;\n\nnamespace tinyxml2\n{\nclass XMLDocument;\nclass XMLElement;\nclass XMLAttribute;\nclass XMLComment;\nclass XMLText;\nclass XMLDeclaration;\nclass XMLUnknown;\nclass XMLPrinter;\n\n/*\n\tA class that wraps strings. Normally stores the start and end\n\tpointers into the XML file itself, and will apply normalization\n\tand entity translation if actually read. Can also store (and memory\n\tmanage) a traditional char[]\n\n    Isn't clear why TINYXML2_LIB is needed; but seems to fix #719\n*/\nclass TINYXML2_LIB StrPair\n{\npublic:\n    enum {\n        NEEDS_ENTITY_PROCESSING\t\t\t= 0x01,\n        NEEDS_NEWLINE_NORMALIZATION\t\t= 0x02,\n        NEEDS_WHITESPACE_COLLAPSING     = 0x04,\n\n        TEXT_ELEMENT\t\t            = NEEDS_ENTITY_PROCESSING | NEEDS_NEWLINE_NORMALIZATION,\n        TEXT_ELEMENT_LEAVE_ENTITIES\t\t= NEEDS_NEWLINE_NORMALIZATION,\n        ATTRIBUTE_NAME\t\t            = 0,\n        ATTRIBUTE_VALUE\t\t            = NEEDS_ENTITY_PROCESSING | NEEDS_NEWLINE_NORMALIZATION,\n        ATTRIBUTE_VALUE_LEAVE_ENTITIES  = NEEDS_NEWLINE_NORMALIZATION,\n        COMMENT\t\t\t\t\t\t\t= NEEDS_NEWLINE_NORMALIZATION\n    };\n\n    StrPair() : _flags( 0 ), _start( 0 ), _end( 0 ) {}\n    ~StrPair();\n\n    void Set( char* start, char* end, int flags ) {\n        TIXMLASSERT( start );\n        TIXMLASSERT( end );\n        Reset();\n        _start  = start;\n        _end    = end;\n        _flags  = flags | NEEDS_FLUSH;\n    }\n\n    const char* GetStr();\n\n    bool Empty() const {\n        return _start == _end;\n    }\n\n    void SetInternedStr( const char* str ) {\n        Reset();\n        _start = const_cast<char*>(str);\n    }\n\n    void SetStr( const char* str, int flags=0 );\n\n    char* ParseText( char* in, const char* endTag, int strFlags, int* curLineNumPtr );\n    char* ParseName( char* in );\n\n    void TransferTo( StrPair* other );\n\tvoid Reset();\n\nprivate:\n    void CollapseWhitespace();\n\n    enum {\n        NEEDS_FLUSH = 0x100,\n        NEEDS_DELETE = 0x200\n    };\n\n    int     _flags;\n    char*   _start;\n    char*   _end;\n\n    StrPair( const StrPair& other );\t// not supported\n    void operator=( const StrPair& other );\t// not supported, use TransferTo()\n};\n\n\n/*\n\tA dynamic array of Plain Old Data. Doesn't support constructors, etc.\n\tHas a small initial memory pool, so that low or no usage will not\n\tcause a call to new/delete\n*/\ntemplate <class T, int INITIAL_SIZE>\nclass DynArray\n{\npublic:\n    DynArray() :\n        _mem( _pool ),\n        _allocated( INITIAL_SIZE ),\n        _size( 0 )\n    {\n    }\n\n    ~DynArray() {\n        if ( _mem != _pool ) {\n            delete [] _mem;\n        }\n    }\n\n    void Clear() {\n        _size = 0;\n    }\n\n    void Push( T t ) {\n        TIXMLASSERT( _size < INT_MAX );\n        EnsureCapacity( _size+1 );\n        _mem[_size] = t;\n        ++_size;\n    }\n\n    T* PushArr( int count ) {\n        TIXMLASSERT( count >= 0 );\n        TIXMLASSERT( _size <= INT_MAX - count );\n        EnsureCapacity( _size+count );\n        T* ret = &_mem[_size];\n        _size += count;\n        return ret;\n    }\n\n    T Pop() {\n        TIXMLASSERT( _size > 0 );\n        --_size;\n        return _mem[_size];\n    }\n\n    void PopArr( int count ) {\n        TIXMLASSERT( _size >= count );\n        _size -= count;\n    }\n\n    bool Empty() const\t\t\t\t\t{\n        return _size == 0;\n    }\n\n    T& operator[](int i)\t\t\t\t{\n        TIXMLASSERT( i>= 0 && i < _size );\n        return _mem[i];\n    }\n\n    const T& operator[](int i) const\t{\n        TIXMLASSERT( i>= 0 && i < _size );\n        return _mem[i];\n    }\n\n    const T& PeekTop() const            {\n        TIXMLASSERT( _size > 0 );\n        return _mem[ _size - 1];\n    }\n\n    int Size() const\t\t\t\t\t{\n        TIXMLASSERT( _size >= 0 );\n        return _size;\n    }\n\n    int Capacity() const\t\t\t\t{\n        TIXMLASSERT( _allocated >= INITIAL_SIZE );\n        return _allocated;\n    }\n\n\tvoid SwapRemove(int i) {\n\t\tTIXMLASSERT(i >= 0 && i < _size);\n\t\tTIXMLASSERT(_size > 0);\n\t\t_mem[i] = _mem[_size - 1];\n\t\t--_size;\n\t}\n\n    const T* Mem() const\t\t\t\t{\n        TIXMLASSERT( _mem );\n        return _mem;\n    }\n\n    T* Mem() {\n        TIXMLASSERT( _mem );\n        return _mem;\n    }\n\nprivate:\n    DynArray( const DynArray& ); // not supported\n    void operator=( const DynArray& ); // not supported\n\n    void EnsureCapacity( int cap ) {\n        TIXMLASSERT( cap > 0 );\n        if ( cap > _allocated ) {\n            TIXMLASSERT( cap <= INT_MAX / 2 );\n            const int newAllocated = cap * 2;\n            T* newMem = new T[newAllocated];\n            TIXMLASSERT( newAllocated >= _size );\n            memcpy( newMem, _mem, sizeof(T)*_size );\t// warning: not using constructors, only works for PODs\n            if ( _mem != _pool ) {\n                delete [] _mem;\n            }\n            _mem = newMem;\n            _allocated = newAllocated;\n        }\n    }\n\n    T*  _mem;\n    T   _pool[INITIAL_SIZE];\n    int _allocated;\t\t// objects allocated\n    int _size;\t\t\t// number objects in use\n};\n\n\n/*\n\tParent virtual class of a pool for fast allocation\n\tand deallocation of objects.\n*/\nclass MemPool\n{\npublic:\n    MemPool() {}\n    virtual ~MemPool() {}\n\n    virtual int ItemSize() const = 0;\n    virtual void* Alloc() = 0;\n    virtual void Free( void* ) = 0;\n    virtual void SetTracked() = 0;\n};\n\n\n/*\n\tTemplate child class to create pools of the correct type.\n*/\ntemplate< int ITEM_SIZE >\nclass MemPoolT : public MemPool\n{\npublic:\n    MemPoolT() : _blockPtrs(), _root(0), _currentAllocs(0), _nAllocs(0), _maxAllocs(0), _nUntracked(0)\t{}\n    ~MemPoolT() {\n        MemPoolT< ITEM_SIZE >::Clear();\n    }\n\n    void Clear() {\n        // Delete the blocks.\n        while( !_blockPtrs.Empty()) {\n            Block* lastBlock = _blockPtrs.Pop();\n            delete lastBlock;\n        }\n        _root = 0;\n        _currentAllocs = 0;\n        _nAllocs = 0;\n        _maxAllocs = 0;\n        _nUntracked = 0;\n    }\n\n    virtual int ItemSize() const\t{\n        return ITEM_SIZE;\n    }\n    int CurrentAllocs() const\t\t{\n        return _currentAllocs;\n    }\n\n    virtual void* Alloc() {\n        if ( !_root ) {\n            // Need a new block.\n            Block* block = new Block();\n            _blockPtrs.Push( block );\n\n            Item* blockItems = block->items;\n            for( int i = 0; i < ITEMS_PER_BLOCK - 1; ++i ) {\n                blockItems[i].next = &(blockItems[i + 1]);\n            }\n            blockItems[ITEMS_PER_BLOCK - 1].next = 0;\n            _root = blockItems;\n        }\n        Item* const result = _root;\n        TIXMLASSERT( result != 0 );\n        _root = _root->next;\n\n        ++_currentAllocs;\n        if ( _currentAllocs > _maxAllocs ) {\n            _maxAllocs = _currentAllocs;\n        }\n        ++_nAllocs;\n        ++_nUntracked;\n        return result;\n    }\n\n    virtual void Free( void* mem ) {\n        if ( !mem ) {\n            return;\n        }\n        --_currentAllocs;\n        Item* item = static_cast<Item*>( mem );\n#ifdef TINYXML2_DEBUG\n        memset( item, 0xfe, sizeof( *item ) );\n#endif\n        item->next = _root;\n        _root = item;\n    }\n    void Trace( const char* name ) {\n        printf( \"Mempool %s watermark=%d [%dk] current=%d size=%d nAlloc=%d blocks=%d\\n\",\n                name, _maxAllocs, _maxAllocs * ITEM_SIZE / 1024, _currentAllocs,\n                ITEM_SIZE, _nAllocs, _blockPtrs.Size() );\n    }\n\n    void SetTracked() {\n        --_nUntracked;\n    }\n\n    int Untracked() const {\n        return _nUntracked;\n    }\n\n\t// This number is perf sensitive. 4k seems like a good tradeoff on my machine.\n\t// The test file is large, 170k.\n\t// Release:\t\tVS2010 gcc(no opt)\n\t//\t\t1k:\t\t4000\n\t//\t\t2k:\t\t4000\n\t//\t\t4k:\t\t3900\t21000\n\t//\t\t16k:\t5200\n\t//\t\t32k:\t4300\n\t//\t\t64k:\t4000\t21000\n    // Declared public because some compilers do not accept to use ITEMS_PER_BLOCK\n    // in private part if ITEMS_PER_BLOCK is private\n    enum { ITEMS_PER_BLOCK = (4 * 1024) / ITEM_SIZE };\n\nprivate:\n    MemPoolT( const MemPoolT& ); // not supported\n    void operator=( const MemPoolT& ); // not supported\n\n    union Item {\n        Item*   next;\n        char    itemData[ITEM_SIZE];\n    };\n    struct Block {\n        Item items[ITEMS_PER_BLOCK];\n    };\n    DynArray< Block*, 10 > _blockPtrs;\n    Item* _root;\n\n    int _currentAllocs;\n    int _nAllocs;\n    int _maxAllocs;\n    int _nUntracked;\n};\n\n\n\n/**\n\tImplements the interface to the \"Visitor pattern\" (see the Accept() method.)\n\tIf you call the Accept() method, it requires being passed a XMLVisitor\n\tclass to handle callbacks. For nodes that contain other nodes (Document, Element)\n\tyou will get called with a VisitEnter/VisitExit pair. Nodes that are always leafs\n\tare simply called with Visit().\n\n\tIf you return 'true' from a Visit method, recursive parsing will continue. If you return\n\tfalse, <b>no children of this node or its siblings</b> will be visited.\n\n\tAll flavors of Visit methods have a default implementation that returns 'true' (continue\n\tvisiting). You need to only override methods that are interesting to you.\n\n\tGenerally Accept() is called on the XMLDocument, although all nodes support visiting.\n\n\tYou should never change the document from a callback.\n\n\t@sa XMLNode::Accept()\n*/\nclass TINYXML2_LIB XMLVisitor\n{\npublic:\n    virtual ~XMLVisitor() {}\n\n    /// Visit a document.\n    virtual bool VisitEnter( const XMLDocument& /*doc*/ )\t\t\t{\n        return true;\n    }\n    /// Visit a document.\n    virtual bool VisitExit( const XMLDocument& /*doc*/ )\t\t\t{\n        return true;\n    }\n\n    /// Visit an element.\n    virtual bool VisitEnter( const XMLElement& /*element*/, const XMLAttribute* /*firstAttribute*/ )\t{\n        return true;\n    }\n    /// Visit an element.\n    virtual bool VisitExit( const XMLElement& /*element*/ )\t\t\t{\n        return true;\n    }\n\n    /// Visit a declaration.\n    virtual bool Visit( const XMLDeclaration& /*declaration*/ )\t\t{\n        return true;\n    }\n    /// Visit a text node.\n    virtual bool Visit( const XMLText& /*text*/ )\t\t\t\t\t{\n        return true;\n    }\n    /// Visit a comment node.\n    virtual bool Visit( const XMLComment& /*comment*/ )\t\t\t\t{\n        return true;\n    }\n    /// Visit an unknown node.\n    virtual bool Visit( const XMLUnknown& /*unknown*/ )\t\t\t\t{\n        return true;\n    }\n};\n\n// WARNING: must match XMLDocument::_errorNames[]\nenum XMLError {\n    XML_SUCCESS = 0,\n    XML_NO_ATTRIBUTE,\n    XML_WRONG_ATTRIBUTE_TYPE,\n    XML_ERROR_FILE_NOT_FOUND,\n    XML_ERROR_FILE_COULD_NOT_BE_OPENED,\n    XML_ERROR_FILE_READ_ERROR,\n    XML_ERROR_PARSING_ELEMENT,\n    XML_ERROR_PARSING_ATTRIBUTE,\n    XML_ERROR_PARSING_TEXT,\n    XML_ERROR_PARSING_CDATA,\n    XML_ERROR_PARSING_COMMENT,\n    XML_ERROR_PARSING_DECLARATION,\n    XML_ERROR_PARSING_UNKNOWN,\n    XML_ERROR_EMPTY_DOCUMENT,\n    XML_ERROR_MISMATCHED_ELEMENT,\n    XML_ERROR_PARSING,\n    XML_CAN_NOT_CONVERT_TEXT,\n    XML_NO_TEXT_NODE,\n\tXML_ELEMENT_DEPTH_EXCEEDED,\n\n\tXML_ERROR_COUNT\n};\n\n\n/*\n\tUtility functionality.\n*/\nclass TINYXML2_LIB XMLUtil\n{\npublic:\n    static const char* SkipWhiteSpace( const char* p, int* curLineNumPtr )\t{\n        TIXMLASSERT( p );\n\n        while( IsWhiteSpace(*p) ) {\n            if (curLineNumPtr && *p == '\\n') {\n                ++(*curLineNumPtr);\n            }\n            ++p;\n        }\n        TIXMLASSERT( p );\n        return p;\n    }\n    static char* SkipWhiteSpace( char* const p, int* curLineNumPtr ) {\n        return const_cast<char*>( SkipWhiteSpace( const_cast<const char*>(p), curLineNumPtr ) );\n    }\n\n    // Anything in the high order range of UTF-8 is assumed to not be whitespace. This isn't\n    // correct, but simple, and usually works.\n    static bool IsWhiteSpace( char p )\t\t\t\t\t{\n        return !IsUTF8Continuation(p) && isspace( static_cast<unsigned char>(p) );\n    }\n\n    inline static bool IsNameStartChar( unsigned char ch ) {\n        if ( ch >= 128 ) {\n            // This is a heuristic guess in attempt to not implement Unicode-aware isalpha()\n            return true;\n        }\n        if ( isalpha( ch ) ) {\n            return true;\n        }\n        return ch == ':' || ch == '_';\n    }\n\n    inline static bool IsNameChar( unsigned char ch ) {\n        return IsNameStartChar( ch )\n               || isdigit( ch )\n               || ch == '.'\n               || ch == '-';\n    }\n\n    inline static bool IsPrefixHex( const char* p) {\n        p = SkipWhiteSpace(p, 0);\n        return p && *p == '0' && ( *(p + 1) == 'x' || *(p + 1) == 'X');\n    }\n\n    inline static bool StringEqual( const char* p, const char* q, int nChar=INT_MAX )  {\n        if ( p == q ) {\n            return true;\n        }\n        TIXMLASSERT( p );\n        TIXMLASSERT( q );\n        TIXMLASSERT( nChar >= 0 );\n        return strncmp( p, q, nChar ) == 0;\n    }\n\n    inline static bool IsUTF8Continuation( const char p ) {\n        return ( p & 0x80 ) != 0;\n    }\n\n    static const char* ReadBOM( const char* p, bool* hasBOM );\n    // p is the starting location,\n    // the UTF-8 value of the entity will be placed in value, and length filled in.\n    static const char* GetCharacterRef( const char* p, char* value, int* length );\n    static void ConvertUTF32ToUTF8( unsigned long input, char* output, int* length );\n\n    // converts primitive types to strings\n    static void ToStr( int v, char* buffer, int bufferSize );\n    static void ToStr( unsigned v, char* buffer, int bufferSize );\n    static void ToStr( bool v, char* buffer, int bufferSize );\n    static void ToStr( float v, char* buffer, int bufferSize );\n    static void ToStr( double v, char* buffer, int bufferSize );\n\tstatic void ToStr(int64_t v, char* buffer, int bufferSize);\n    static void ToStr(uint64_t v, char* buffer, int bufferSize);\n\n    // converts strings to primitive types\n    static bool\tToInt( const char* str, int* value );\n    static bool ToUnsigned( const char* str, unsigned* value );\n    static bool\tToBool( const char* str, bool* value );\n    static bool\tToFloat( const char* str, float* value );\n    static bool ToDouble( const char* str, double* value );\n\tstatic bool ToInt64(const char* str, int64_t* value);\n    static bool ToUnsigned64(const char* str, uint64_t* value);\n\t// Changes what is serialized for a boolean value.\n\t// Default to \"true\" and \"false\". Shouldn't be changed\n\t// unless you have a special testing or compatibility need.\n\t// Be careful: static, global, & not thread safe.\n\t// Be sure to set static const memory as parameters.\n\tstatic void SetBoolSerialization(const char* writeTrue, const char* writeFalse);\n\nprivate:\n\tstatic const char* writeBoolTrue;\n\tstatic const char* writeBoolFalse;\n};\n\n\n/** XMLNode is a base class for every object that is in the\n\tXML Document Object Model (DOM), except XMLAttributes.\n\tNodes have siblings, a parent, and children which can\n\tbe navigated. A node is always in a XMLDocument.\n\tThe type of a XMLNode can be queried, and it can\n\tbe cast to its more defined type.\n\n\tA XMLDocument allocates memory for all its Nodes.\n\tWhen the XMLDocument gets deleted, all its Nodes\n\twill also be deleted.\n\n\t@verbatim\n\tA Document can contain:\tElement\t(container or leaf)\n\t\t\t\t\t\t\tComment (leaf)\n\t\t\t\t\t\t\tUnknown (leaf)\n\t\t\t\t\t\t\tDeclaration( leaf )\n\n\tAn Element can contain:\tElement (container or leaf)\n\t\t\t\t\t\t\tText\t(leaf)\n\t\t\t\t\t\t\tAttributes (not on tree)\n\t\t\t\t\t\t\tComment (leaf)\n\t\t\t\t\t\t\tUnknown (leaf)\n\n\t@endverbatim\n*/\nclass TINYXML2_LIB XMLNode\n{\n    friend class XMLDocument;\n    friend class XMLElement;\npublic:\n\n    /// Get the XMLDocument that owns this XMLNode.\n    const XMLDocument* GetDocument() const\t{\n        TIXMLASSERT( _document );\n        return _document;\n    }\n    /// Get the XMLDocument that owns this XMLNode.\n    XMLDocument* GetDocument()\t\t\t\t{\n        TIXMLASSERT( _document );\n        return _document;\n    }\n\n    /// Safely cast to an Element, or null.\n    virtual XMLElement*\t\tToElement()\t\t{\n        return 0;\n    }\n    /// Safely cast to Text, or null.\n    virtual XMLText*\t\tToText()\t\t{\n        return 0;\n    }\n    /// Safely cast to a Comment, or null.\n    virtual XMLComment*\t\tToComment()\t\t{\n        return 0;\n    }\n    /// Safely cast to a Document, or null.\n    virtual XMLDocument*\tToDocument()\t{\n        return 0;\n    }\n    /// Safely cast to a Declaration, or null.\n    virtual XMLDeclaration*\tToDeclaration()\t{\n        return 0;\n    }\n    /// Safely cast to an Unknown, or null.\n    virtual XMLUnknown*\t\tToUnknown()\t\t{\n        return 0;\n    }\n\n    virtual const XMLElement*\t\tToElement() const\t\t{\n        return 0;\n    }\n    virtual const XMLText*\t\t\tToText() const\t\t\t{\n        return 0;\n    }\n    virtual const XMLComment*\t\tToComment() const\t\t{\n        return 0;\n    }\n    virtual const XMLDocument*\t\tToDocument() const\t\t{\n        return 0;\n    }\n    virtual const XMLDeclaration*\tToDeclaration() const\t{\n        return 0;\n    }\n    virtual const XMLUnknown*\t\tToUnknown() const\t\t{\n        return 0;\n    }\n\n    /** The meaning of 'value' changes for the specific type.\n    \t@verbatim\n    \tDocument:\tempty (NULL is returned, not an empty string)\n    \tElement:\tname of the element\n    \tComment:\tthe comment text\n    \tUnknown:\tthe tag contents\n    \tText:\t\tthe text string\n    \t@endverbatim\n    */\n    const char* Value() const;\n\n    /** Set the Value of an XML node.\n    \t@sa Value()\n    */\n    void SetValue( const char* val, bool staticMem=false );\n\n    /// Gets the line number the node is in, if the document was parsed from a file.\n    int GetLineNum() const { return _parseLineNum; }\n\n    /// Get the parent of this node on the DOM.\n    const XMLNode*\tParent() const\t\t\t{\n        return _parent;\n    }\n\n    XMLNode* Parent()\t\t\t\t\t\t{\n        return _parent;\n    }\n\n    /// Returns true if this node has no children.\n    bool NoChildren() const\t\t\t\t\t{\n        return !_firstChild;\n    }\n\n    /// Get the first child node, or null if none exists.\n    const XMLNode*  FirstChild() const\t\t{\n        return _firstChild;\n    }\n\n    XMLNode*\t\tFirstChild()\t\t\t{\n        return _firstChild;\n    }\n\n    /** Get the first child element, or optionally the first child\n        element with the specified name.\n    */\n    const XMLElement* FirstChildElement( const char* name = 0 ) const;\n\n    XMLElement* FirstChildElement( const char* name = 0 )\t{\n        return const_cast<XMLElement*>(const_cast<const XMLNode*>(this)->FirstChildElement( name ));\n    }\n\n    /// Get the last child node, or null if none exists.\n    const XMLNode*\tLastChild() const\t\t\t\t\t\t{\n        return _lastChild;\n    }\n\n    XMLNode*\t\tLastChild()\t\t\t\t\t\t\t\t{\n        return _lastChild;\n    }\n\n    /** Get the last child element or optionally the last child\n        element with the specified name.\n    */\n    const XMLElement* LastChildElement( const char* name = 0 ) const;\n\n    XMLElement* LastChildElement( const char* name = 0 )\t{\n        return const_cast<XMLElement*>(const_cast<const XMLNode*>(this)->LastChildElement(name) );\n    }\n\n    /// Get the previous (left) sibling node of this node.\n    const XMLNode*\tPreviousSibling() const\t\t\t\t\t{\n        return _prev;\n    }\n\n    XMLNode*\tPreviousSibling()\t\t\t\t\t\t\t{\n        return _prev;\n    }\n\n    /// Get the previous (left) sibling element of this node, with an optionally supplied name.\n    const XMLElement*\tPreviousSiblingElement( const char* name = 0 ) const ;\n\n    XMLElement*\tPreviousSiblingElement( const char* name = 0 ) {\n        return const_cast<XMLElement*>(const_cast<const XMLNode*>(this)->PreviousSiblingElement( name ) );\n    }\n\n    /// Get the next (right) sibling node of this node.\n    const XMLNode*\tNextSibling() const\t\t\t\t\t\t{\n        return _next;\n    }\n\n    XMLNode*\tNextSibling()\t\t\t\t\t\t\t\t{\n        return _next;\n    }\n\n    /// Get the next (right) sibling element of this node, with an optionally supplied name.\n    const XMLElement*\tNextSiblingElement( const char* name = 0 ) const;\n\n    XMLElement*\tNextSiblingElement( const char* name = 0 )\t{\n        return const_cast<XMLElement*>(const_cast<const XMLNode*>(this)->NextSiblingElement( name ) );\n    }\n\n    /**\n    \tAdd a child node as the last (right) child.\n\t\tIf the child node is already part of the document,\n\t\tit is moved from its old location to the new location.\n\t\tReturns the addThis argument or 0 if the node does not\n\t\tbelong to the same document.\n    */\n    XMLNode* InsertEndChild( XMLNode* addThis );\n\n    XMLNode* LinkEndChild( XMLNode* addThis )\t{\n        return InsertEndChild( addThis );\n    }\n    /**\n    \tAdd a child node as the first (left) child.\n\t\tIf the child node is already part of the document,\n\t\tit is moved from its old location to the new location.\n\t\tReturns the addThis argument or 0 if the node does not\n\t\tbelong to the same document.\n    */\n    XMLNode* InsertFirstChild( XMLNode* addThis );\n    /**\n    \tAdd a node after the specified child node.\n\t\tIf the child node is already part of the document,\n\t\tit is moved from its old location to the new location.\n\t\tReturns the addThis argument or 0 if the afterThis node\n\t\tis not a child of this node, or if the node does not\n\t\tbelong to the same document.\n    */\n    XMLNode* InsertAfterChild( XMLNode* afterThis, XMLNode* addThis );\n\n    /**\n    \tDelete all the children of this node.\n    */\n    void DeleteChildren();\n\n    /**\n    \tDelete a child of this node.\n    */\n    void DeleteChild( XMLNode* node );\n\n    /**\n    \tMake a copy of this node, but not its children.\n    \tYou may pass in a Document pointer that will be\n    \tthe owner of the new Node. If the 'document' is\n    \tnull, then the node returned will be allocated\n    \tfrom the current Document. (this->GetDocument())\n\n    \tNote: if called on a XMLDocument, this will return null.\n    */\n    virtual XMLNode* ShallowClone( XMLDocument* document ) const = 0;\n\n\t/**\n\t\tMake a copy of this node and all its children.\n\n\t\tIf the 'target' is null, then the nodes will\n\t\tbe allocated in the current document. If 'target'\n        is specified, the memory will be allocated is the\n        specified XMLDocument.\n\n\t\tNOTE: This is probably not the correct tool to\n\t\tcopy a document, since XMLDocuments can have multiple\n\t\ttop level XMLNodes. You probably want to use\n        XMLDocument::DeepCopy()\n\t*/\n\tXMLNode* DeepClone( XMLDocument* target ) const;\n\n    /**\n    \tTest if 2 nodes are the same, but don't test children.\n    \tThe 2 nodes do not need to be in the same Document.\n\n    \tNote: if called on a XMLDocument, this will return false.\n    */\n    virtual bool ShallowEqual( const XMLNode* compare ) const = 0;\n\n    /** Accept a hierarchical visit of the nodes in the TinyXML-2 DOM. Every node in the\n    \tXML tree will be conditionally visited and the host will be called back\n    \tvia the XMLVisitor interface.\n\n    \tThis is essentially a SAX interface for TinyXML-2. (Note however it doesn't re-parse\n    \tthe XML for the callbacks, so the performance of TinyXML-2 is unchanged by using this\n    \tinterface versus any other.)\n\n    \tThe interface has been based on ideas from:\n\n    \t- http://www.saxproject.org/\n    \t- http://c2.com/cgi/wiki?HierarchicalVisitorPattern\n\n    \tWhich are both good references for \"visiting\".\n\n    \tAn example of using Accept():\n    \t@verbatim\n    \tXMLPrinter printer;\n    \ttinyxmlDoc.Accept( &printer );\n    \tconst char* xmlcstr = printer.CStr();\n    \t@endverbatim\n    */\n    virtual bool Accept( XMLVisitor* visitor ) const = 0;\n\n\t/**\n\t\tSet user data into the XMLNode. TinyXML-2 in\n\t\tno way processes or interprets user data.\n\t\tIt is initially 0.\n\t*/\n\tvoid SetUserData(void* userData)\t{ _userData = userData; }\n\n\t/**\n\t\tGet user data set into the XMLNode. TinyXML-2 in\n\t\tno way processes or interprets user data.\n\t\tIt is initially 0.\n\t*/\n\tvoid* GetUserData() const\t\t\t{ return _userData; }\n\nprotected:\n    explicit XMLNode( XMLDocument* );\n    virtual ~XMLNode();\n\n    virtual char* ParseDeep( char* p, StrPair* parentEndTag, int* curLineNumPtr);\n\n    XMLDocument*\t_document;\n    XMLNode*\t\t_parent;\n    mutable StrPair\t_value;\n    int             _parseLineNum;\n\n    XMLNode*\t\t_firstChild;\n    XMLNode*\t\t_lastChild;\n\n    XMLNode*\t\t_prev;\n    XMLNode*\t\t_next;\n\n\tvoid*\t\t\t_userData;\n\nprivate:\n    MemPool*\t\t_memPool;\n    void Unlink( XMLNode* child );\n    static void DeleteNode( XMLNode* node );\n    void InsertChildPreamble( XMLNode* insertThis ) const;\n    const XMLElement* ToElementWithName( const char* name ) const;\n\n    XMLNode( const XMLNode& );\t// not supported\n    XMLNode& operator=( const XMLNode& );\t// not supported\n};\n\n\n/** XML text.\n\n\tNote that a text node can have child element nodes, for example:\n\t@verbatim\n\t<root>This is <b>bold</b></root>\n\t@endverbatim\n\n\tA text node can have 2 ways to output the next. \"normal\" output\n\tand CDATA. It will default to the mode it was parsed from the XML file and\n\tyou generally want to leave it alone, but you can change the output mode with\n\tSetCData() and query it with CData().\n*/\nclass TINYXML2_LIB XMLText : public XMLNode\n{\n    friend class XMLDocument;\npublic:\n    virtual bool Accept( XMLVisitor* visitor ) const;\n\n    virtual XMLText* ToText()\t\t\t{\n        return this;\n    }\n    virtual const XMLText* ToText() const\t{\n        return this;\n    }\n\n    /// Declare whether this should be CDATA or standard text.\n    void SetCData( bool isCData )\t\t\t{\n        _isCData = isCData;\n    }\n    /// Returns true if this is a CDATA text element.\n    bool CData() const\t\t\t\t\t\t{\n        return _isCData;\n    }\n\n    virtual XMLNode* ShallowClone( XMLDocument* document ) const;\n    virtual bool ShallowEqual( const XMLNode* compare ) const;\n\nprotected:\n    explicit XMLText( XMLDocument* doc )\t: XMLNode( doc ), _isCData( false )\t{}\n    virtual ~XMLText()\t\t\t\t\t\t\t\t\t\t\t\t{}\n\n    char* ParseDeep( char* p, StrPair* parentEndTag, int* curLineNumPtr );\n\nprivate:\n    bool _isCData;\n\n    XMLText( const XMLText& );\t// not supported\n    XMLText& operator=( const XMLText& );\t// not supported\n};\n\n\n/** An XML Comment. */\nclass TINYXML2_LIB XMLComment : public XMLNode\n{\n    friend class XMLDocument;\npublic:\n    virtual XMLComment*\tToComment()\t\t\t\t\t{\n        return this;\n    }\n    virtual const XMLComment* ToComment() const\t\t{\n        return this;\n    }\n\n    virtual bool Accept( XMLVisitor* visitor ) const;\n\n    virtual XMLNode* ShallowClone( XMLDocument* document ) const;\n    virtual bool ShallowEqual( const XMLNode* compare ) const;\n\nprotected:\n    explicit XMLComment( XMLDocument* doc );\n    virtual ~XMLComment();\n\n    char* ParseDeep( char* p, StrPair* parentEndTag, int* curLineNumPtr);\n\nprivate:\n    XMLComment( const XMLComment& );\t// not supported\n    XMLComment& operator=( const XMLComment& );\t// not supported\n};\n\n\n/** In correct XML the declaration is the first entry in the file.\n\t@verbatim\n\t\t<?xml version=\"1.0\" standalone=\"yes\"?>\n\t@endverbatim\n\n\tTinyXML-2 will happily read or write files without a declaration,\n\thowever.\n\n\tThe text of the declaration isn't interpreted. It is parsed\n\tand written as a string.\n*/\nclass TINYXML2_LIB XMLDeclaration : public XMLNode\n{\n    friend class XMLDocument;\npublic:\n    virtual XMLDeclaration*\tToDeclaration()\t\t\t\t\t{\n        return this;\n    }\n    virtual const XMLDeclaration* ToDeclaration() const\t\t{\n        return this;\n    }\n\n    virtual bool Accept( XMLVisitor* visitor ) const;\n\n    virtual XMLNode* ShallowClone( XMLDocument* document ) const;\n    virtual bool ShallowEqual( const XMLNode* compare ) const;\n\nprotected:\n    explicit XMLDeclaration( XMLDocument* doc );\n    virtual ~XMLDeclaration();\n\n    char* ParseDeep( char* p, StrPair* parentEndTag, int* curLineNumPtr );\n\nprivate:\n    XMLDeclaration( const XMLDeclaration& );\t// not supported\n    XMLDeclaration& operator=( const XMLDeclaration& );\t// not supported\n};\n\n\n/** Any tag that TinyXML-2 doesn't recognize is saved as an\n\tunknown. It is a tag of text, but should not be modified.\n\tIt will be written back to the XML, unchanged, when the file\n\tis saved.\n\n\tDTD tags get thrown into XMLUnknowns.\n*/\nclass TINYXML2_LIB XMLUnknown : public XMLNode\n{\n    friend class XMLDocument;\npublic:\n    virtual XMLUnknown*\tToUnknown()\t\t\t\t\t{\n        return this;\n    }\n    virtual const XMLUnknown* ToUnknown() const\t\t{\n        return this;\n    }\n\n    virtual bool Accept( XMLVisitor* visitor ) const;\n\n    virtual XMLNode* ShallowClone( XMLDocument* document ) const;\n    virtual bool ShallowEqual( const XMLNode* compare ) const;\n\nprotected:\n    explicit XMLUnknown( XMLDocument* doc );\n    virtual ~XMLUnknown();\n\n    char* ParseDeep( char* p, StrPair* parentEndTag, int* curLineNumPtr );\n\nprivate:\n    XMLUnknown( const XMLUnknown& );\t// not supported\n    XMLUnknown& operator=( const XMLUnknown& );\t// not supported\n};\n\n\n\n/** An attribute is a name-value pair. Elements have an arbitrary\n\tnumber of attributes, each with a unique name.\n\n\t@note The attributes are not XMLNodes. You may only query the\n\tNext() attribute in a list.\n*/\nclass TINYXML2_LIB XMLAttribute\n{\n    friend class XMLElement;\npublic:\n    /// The name of the attribute.\n    const char* Name() const;\n\n    /// The value of the attribute.\n    const char* Value() const;\n\n    /// Gets the line number the attribute is in, if the document was parsed from a file.\n    int GetLineNum() const { return _parseLineNum; }\n\n    /// The next attribute in the list.\n    const XMLAttribute* Next() const {\n        return _next;\n    }\n\n    /** IntValue interprets the attribute as an integer, and returns the value.\n        If the value isn't an integer, 0 will be returned. There is no error checking;\n    \tuse QueryIntValue() if you need error checking.\n    */\n\tint\tIntValue() const {\n\t\tint i = 0;\n\t\tQueryIntValue(&i);\n\t\treturn i;\n\t}\n\n\tint64_t Int64Value() const {\n\t\tint64_t i = 0;\n\t\tQueryInt64Value(&i);\n\t\treturn i;\n\t}\n\n    uint64_t Unsigned64Value() const {\n        uint64_t i = 0;\n        QueryUnsigned64Value(&i);\n        return i;\n    }\n\n    /// Query as an unsigned integer. See IntValue()\n    unsigned UnsignedValue() const\t\t\t{\n        unsigned i=0;\n        QueryUnsignedValue( &i );\n        return i;\n    }\n    /// Query as a boolean. See IntValue()\n    bool\t BoolValue() const\t\t\t\t{\n        bool b=false;\n        QueryBoolValue( &b );\n        return b;\n    }\n    /// Query as a double. See IntValue()\n    double \t DoubleValue() const\t\t\t{\n        double d=0;\n        QueryDoubleValue( &d );\n        return d;\n    }\n    /// Query as a float. See IntValue()\n    float\t FloatValue() const\t\t\t\t{\n        float f=0;\n        QueryFloatValue( &f );\n        return f;\n    }\n\n    /** QueryIntValue interprets the attribute as an integer, and returns the value\n    \tin the provided parameter. The function will return XML_SUCCESS on success,\n    \tand XML_WRONG_ATTRIBUTE_TYPE if the conversion is not successful.\n    */\n    XMLError QueryIntValue( int* value ) const;\n    /// See QueryIntValue\n    XMLError QueryUnsignedValue( unsigned int* value ) const;\n\t/// See QueryIntValue\n\tXMLError QueryInt64Value(int64_t* value) const;\n    /// See QueryIntValue\n    XMLError QueryUnsigned64Value(uint64_t* value) const;\n\t/// See QueryIntValue\n    XMLError QueryBoolValue( bool* value ) const;\n    /// See QueryIntValue\n    XMLError QueryDoubleValue( double* value ) const;\n    /// See QueryIntValue\n    XMLError QueryFloatValue( float* value ) const;\n\n    /// Set the attribute to a string value.\n    void SetAttribute( const char* value );\n    /// Set the attribute to value.\n    void SetAttribute( int value );\n    /// Set the attribute to value.\n    void SetAttribute( unsigned value );\n\t/// Set the attribute to value.\n\tvoid SetAttribute(int64_t value);\n    /// Set the attribute to value.\n    void SetAttribute(uint64_t value);\n    /// Set the attribute to value.\n    void SetAttribute( bool value );\n    /// Set the attribute to value.\n    void SetAttribute( double value );\n    /// Set the attribute to value.\n    void SetAttribute( float value );\n\nprivate:\n    enum { BUF_SIZE = 200 };\n\n    XMLAttribute() : _name(), _value(),_parseLineNum( 0 ), _next( 0 ), _memPool( 0 ) {}\n    virtual ~XMLAttribute()\t{}\n\n    XMLAttribute( const XMLAttribute& );\t// not supported\n    void operator=( const XMLAttribute& );\t// not supported\n    void SetName( const char* name );\n\n    char* ParseDeep( char* p, bool processEntities, int* curLineNumPtr );\n\n    mutable StrPair _name;\n    mutable StrPair _value;\n    int             _parseLineNum;\n    XMLAttribute*   _next;\n    MemPool*        _memPool;\n};\n\n\n/** The element is a container class. It has a value, the element name,\n\tand can contain other elements, text, comments, and unknowns.\n\tElements also contain an arbitrary number of attributes.\n*/\nclass TINYXML2_LIB XMLElement : public XMLNode\n{\n    friend class XMLDocument;\npublic:\n    /// Get the name of an element (which is the Value() of the node.)\n    const char* Name() const\t\t{\n        return Value();\n    }\n    /// Set the name of the element.\n    void SetName( const char* str, bool staticMem=false )\t{\n        SetValue( str, staticMem );\n    }\n\n    virtual XMLElement* ToElement()\t\t\t\t{\n        return this;\n    }\n    virtual const XMLElement* ToElement() const {\n        return this;\n    }\n    virtual bool Accept( XMLVisitor* visitor ) const;\n\n    /** Given an attribute name, Attribute() returns the value\n    \tfor the attribute of that name, or null if none\n    \texists. For example:\n\n    \t@verbatim\n    \tconst char* value = ele->Attribute( \"foo\" );\n    \t@endverbatim\n\n    \tThe 'value' parameter is normally null. However, if specified,\n    \tthe attribute will only be returned if the 'name' and 'value'\n    \tmatch. This allow you to write code:\n\n    \t@verbatim\n    \tif ( ele->Attribute( \"foo\", \"bar\" ) ) callFooIsBar();\n    \t@endverbatim\n\n    \trather than:\n    \t@verbatim\n    \tif ( ele->Attribute( \"foo\" ) ) {\n    \t\tif ( strcmp( ele->Attribute( \"foo\" ), \"bar\" ) == 0 ) callFooIsBar();\n    \t}\n    \t@endverbatim\n    */\n    const char* Attribute( const char* name, const char* value=0 ) const;\n\n    /** Given an attribute name, IntAttribute() returns the value\n    \tof the attribute interpreted as an integer. The default\n        value will be returned if the attribute isn't present,\n        or if there is an error. (For a method with error\n    \tchecking, see QueryIntAttribute()).\n    */\n\tint IntAttribute(const char* name, int defaultValue = 0) const;\n    /// See IntAttribute()\n\tunsigned UnsignedAttribute(const char* name, unsigned defaultValue = 0) const;\n\t/// See IntAttribute()\n\tint64_t Int64Attribute(const char* name, int64_t defaultValue = 0) const;\n    /// See IntAttribute()\n    uint64_t Unsigned64Attribute(const char* name, uint64_t defaultValue = 0) const;\n\t/// See IntAttribute()\n\tbool BoolAttribute(const char* name, bool defaultValue = false) const;\n    /// See IntAttribute()\n\tdouble DoubleAttribute(const char* name, double defaultValue = 0) const;\n    /// See IntAttribute()\n\tfloat FloatAttribute(const char* name, float defaultValue = 0) const;\n\n    /** Given an attribute name, QueryIntAttribute() returns\n    \tXML_SUCCESS, XML_WRONG_ATTRIBUTE_TYPE if the conversion\n    \tcan't be performed, or XML_NO_ATTRIBUTE if the attribute\n    \tdoesn't exist. If successful, the result of the conversion\n    \twill be written to 'value'. If not successful, nothing will\n    \tbe written to 'value'. This allows you to provide default\n    \tvalue:\n\n    \t@verbatim\n    \tint value = 10;\n    \tQueryIntAttribute( \"foo\", &value );\t\t// if \"foo\" isn't found, value will still be 10\n    \t@endverbatim\n    */\n    XMLError QueryIntAttribute( const char* name, int* value ) const\t\t\t\t{\n        const XMLAttribute* a = FindAttribute( name );\n        if ( !a ) {\n            return XML_NO_ATTRIBUTE;\n        }\n        return a->QueryIntValue( value );\n    }\n\n\t/// See QueryIntAttribute()\n    XMLError QueryUnsignedAttribute( const char* name, unsigned int* value ) const\t{\n        const XMLAttribute* a = FindAttribute( name );\n        if ( !a ) {\n            return XML_NO_ATTRIBUTE;\n        }\n        return a->QueryUnsignedValue( value );\n    }\n\n\t/// See QueryIntAttribute()\n\tXMLError QueryInt64Attribute(const char* name, int64_t* value) const {\n\t\tconst XMLAttribute* a = FindAttribute(name);\n\t\tif (!a) {\n\t\t\treturn XML_NO_ATTRIBUTE;\n\t\t}\n\t\treturn a->QueryInt64Value(value);\n\t}\n\n    /// See QueryIntAttribute()\n    XMLError QueryUnsigned64Attribute(const char* name, uint64_t* value) const {\n        const XMLAttribute* a = FindAttribute(name);\n        if(!a) {\n            return XML_NO_ATTRIBUTE;\n        }\n        return a->QueryUnsigned64Value(value);\n    }\n\n\t/// See QueryIntAttribute()\n    XMLError QueryBoolAttribute( const char* name, bool* value ) const\t\t\t\t{\n        const XMLAttribute* a = FindAttribute( name );\n        if ( !a ) {\n            return XML_NO_ATTRIBUTE;\n        }\n        return a->QueryBoolValue( value );\n    }\n    /// See QueryIntAttribute()\n    XMLError QueryDoubleAttribute( const char* name, double* value ) const\t\t\t{\n        const XMLAttribute* a = FindAttribute( name );\n        if ( !a ) {\n            return XML_NO_ATTRIBUTE;\n        }\n        return a->QueryDoubleValue( value );\n    }\n    /// See QueryIntAttribute()\n    XMLError QueryFloatAttribute( const char* name, float* value ) const\t\t\t{\n        const XMLAttribute* a = FindAttribute( name );\n        if ( !a ) {\n            return XML_NO_ATTRIBUTE;\n        }\n        return a->QueryFloatValue( value );\n    }\n\n\t/// See QueryIntAttribute()\n\tXMLError QueryStringAttribute(const char* name, const char** value) const {\n\t\tconst XMLAttribute* a = FindAttribute(name);\n\t\tif (!a) {\n\t\t\treturn XML_NO_ATTRIBUTE;\n\t\t}\n\t\t*value = a->Value();\n\t\treturn XML_SUCCESS;\n\t}\n\n\n\n    /** Given an attribute name, QueryAttribute() returns\n    \tXML_SUCCESS, XML_WRONG_ATTRIBUTE_TYPE if the conversion\n    \tcan't be performed, or XML_NO_ATTRIBUTE if the attribute\n    \tdoesn't exist. It is overloaded for the primitive types,\n\t\tand is a generally more convenient replacement of\n\t\tQueryIntAttribute() and related functions.\n\n\t\tIf successful, the result of the conversion\n    \twill be written to 'value'. If not successful, nothing will\n    \tbe written to 'value'. This allows you to provide default\n    \tvalue:\n\n    \t@verbatim\n    \tint value = 10;\n    \tQueryAttribute( \"foo\", &value );\t\t// if \"foo\" isn't found, value will still be 10\n    \t@endverbatim\n    */\n\tXMLError QueryAttribute( const char* name, int* value ) const {\n\t\treturn QueryIntAttribute( name, value );\n\t}\n\n\tXMLError QueryAttribute( const char* name, unsigned int* value ) const {\n\t\treturn QueryUnsignedAttribute( name, value );\n\t}\n\n\tXMLError QueryAttribute(const char* name, int64_t* value) const {\n\t\treturn QueryInt64Attribute(name, value);\n\t}\n\n    XMLError QueryAttribute(const char* name, uint64_t* value) const {\n        return QueryUnsigned64Attribute(name, value);\n    }\n\n    XMLError QueryAttribute( const char* name, bool* value ) const {\n\t\treturn QueryBoolAttribute( name, value );\n\t}\n\n\tXMLError QueryAttribute( const char* name, double* value ) const {\n\t\treturn QueryDoubleAttribute( name, value );\n\t}\n\n\tXMLError QueryAttribute( const char* name, float* value ) const {\n\t\treturn QueryFloatAttribute( name, value );\n\t}\n\n\tXMLError QueryAttribute(const char* name, const char** value) const {\n\t\treturn QueryStringAttribute(name, value);\n\t}\n\n\t/// Sets the named attribute to value.\n    void SetAttribute( const char* name, const char* value )\t{\n        XMLAttribute* a = FindOrCreateAttribute( name );\n        a->SetAttribute( value );\n    }\n    /// Sets the named attribute to value.\n    void SetAttribute( const char* name, int value )\t\t\t{\n        XMLAttribute* a = FindOrCreateAttribute( name );\n        a->SetAttribute( value );\n    }\n    /// Sets the named attribute to value.\n    void SetAttribute( const char* name, unsigned value )\t\t{\n        XMLAttribute* a = FindOrCreateAttribute( name );\n        a->SetAttribute( value );\n    }\n\n\t/// Sets the named attribute to value.\n\tvoid SetAttribute(const char* name, int64_t value) {\n\t\tXMLAttribute* a = FindOrCreateAttribute(name);\n\t\ta->SetAttribute(value);\n\t}\n\n    /// Sets the named attribute to value.\n    void SetAttribute(const char* name, uint64_t value) {\n        XMLAttribute* a = FindOrCreateAttribute(name);\n        a->SetAttribute(value);\n    }\n\n    /// Sets the named attribute to value.\n    void SetAttribute( const char* name, bool value )\t\t\t{\n        XMLAttribute* a = FindOrCreateAttribute( name );\n        a->SetAttribute( value );\n    }\n    /// Sets the named attribute to value.\n    void SetAttribute( const char* name, double value )\t\t{\n        XMLAttribute* a = FindOrCreateAttribute( name );\n        a->SetAttribute( value );\n    }\n    /// Sets the named attribute to value.\n    void SetAttribute( const char* name, float value )\t\t{\n        XMLAttribute* a = FindOrCreateAttribute( name );\n        a->SetAttribute( value );\n    }\n\n    /**\n    \tDelete an attribute.\n    */\n    void DeleteAttribute( const char* name );\n\n    /// Return the first attribute in the list.\n    const XMLAttribute* FirstAttribute() const {\n        return _rootAttribute;\n    }\n    /// Query a specific attribute in the list.\n    const XMLAttribute* FindAttribute( const char* name ) const;\n\n    /** Convenience function for easy access to the text inside an element. Although easy\n    \tand concise, GetText() is limited compared to getting the XMLText child\n    \tand accessing it directly.\n\n    \tIf the first child of 'this' is a XMLText, the GetText()\n    \treturns the character string of the Text node, else null is returned.\n\n    \tThis is a convenient method for getting the text of simple contained text:\n    \t@verbatim\n    \t<foo>This is text</foo>\n    \t\tconst char* str = fooElement->GetText();\n    \t@endverbatim\n\n    \t'str' will be a pointer to \"This is text\".\n\n    \tNote that this function can be misleading. If the element foo was created from\n    \tthis XML:\n    \t@verbatim\n    \t\t<foo><b>This is text</b></foo>\n    \t@endverbatim\n\n    \tthen the value of str would be null. The first child node isn't a text node, it is\n    \tanother element. From this XML:\n    \t@verbatim\n    \t\t<foo>This is <b>text</b></foo>\n    \t@endverbatim\n    \tGetText() will return \"This is \".\n    */\n    const char* GetText() const;\n\n    /** Convenience function for easy access to the text inside an element. Although easy\n    \tand concise, SetText() is limited compared to creating an XMLText child\n    \tand mutating it directly.\n\n    \tIf the first child of 'this' is a XMLText, SetText() sets its value to\n\t\tthe given string, otherwise it will create a first child that is an XMLText.\n\n    \tThis is a convenient method for setting the text of simple contained text:\n    \t@verbatim\n    \t<foo>This is text</foo>\n    \t\tfooElement->SetText( \"Hullaballoo!\" );\n     \t<foo>Hullaballoo!</foo>\n\t\t@endverbatim\n\n    \tNote that this function can be misleading. If the element foo was created from\n    \tthis XML:\n    \t@verbatim\n    \t\t<foo><b>This is text</b></foo>\n    \t@endverbatim\n\n    \tthen it will not change \"This is text\", but rather prefix it with a text element:\n    \t@verbatim\n    \t\t<foo>Hullaballoo!<b>This is text</b></foo>\n    \t@endverbatim\n\n\t\tFor this XML:\n    \t@verbatim\n    \t\t<foo />\n    \t@endverbatim\n    \tSetText() will generate\n    \t@verbatim\n    \t\t<foo>Hullaballoo!</foo>\n    \t@endverbatim\n    */\n\tvoid SetText( const char* inText );\n    /// Convenience method for setting text inside an element. See SetText() for important limitations.\n    void SetText( int value );\n    /// Convenience method for setting text inside an element. See SetText() for important limitations.\n    void SetText( unsigned value );\n\t/// Convenience method for setting text inside an element. See SetText() for important limitations.\n\tvoid SetText(int64_t value);\n    /// Convenience method for setting text inside an element. See SetText() for important limitations.\n    void SetText(uint64_t value);\n\t/// Convenience method for setting text inside an element. See SetText() for important limitations.\n    void SetText( bool value );\n    /// Convenience method for setting text inside an element. See SetText() for important limitations.\n    void SetText( double value );\n    /// Convenience method for setting text inside an element. See SetText() for important limitations.\n    void SetText( float value );\n\n    /**\n    \tConvenience method to query the value of a child text node. This is probably best\n    \tshown by example. Given you have a document is this form:\n    \t@verbatim\n    \t\t<point>\n    \t\t\t<x>1</x>\n    \t\t\t<y>1.4</y>\n    \t\t</point>\n    \t@endverbatim\n\n    \tThe QueryIntText() and similar functions provide a safe and easier way to get to the\n    \t\"value\" of x and y.\n\n    \t@verbatim\n    \t\tint x = 0;\n    \t\tfloat y = 0;\t// types of x and y are contrived for example\n    \t\tconst XMLElement* xElement = pointElement->FirstChildElement( \"x\" );\n    \t\tconst XMLElement* yElement = pointElement->FirstChildElement( \"y\" );\n    \t\txElement->QueryIntText( &x );\n    \t\tyElement->QueryFloatText( &y );\n    \t@endverbatim\n\n    \t@returns XML_SUCCESS (0) on success, XML_CAN_NOT_CONVERT_TEXT if the text cannot be converted\n    \t\t\t to the requested type, and XML_NO_TEXT_NODE if there is no child text to query.\n\n    */\n    XMLError QueryIntText( int* ival ) const;\n    /// See QueryIntText()\n    XMLError QueryUnsignedText( unsigned* uval ) const;\n\t/// See QueryIntText()\n\tXMLError QueryInt64Text(int64_t* uval) const;\n\t/// See QueryIntText()\n\tXMLError QueryUnsigned64Text(uint64_t* uval) const;\n\t/// See QueryIntText()\n    XMLError QueryBoolText( bool* bval ) const;\n    /// See QueryIntText()\n    XMLError QueryDoubleText( double* dval ) const;\n    /// See QueryIntText()\n    XMLError QueryFloatText( float* fval ) const;\n\n\tint IntText(int defaultValue = 0) const;\n\n\t/// See QueryIntText()\n\tunsigned UnsignedText(unsigned defaultValue = 0) const;\n\t/// See QueryIntText()\n\tint64_t Int64Text(int64_t defaultValue = 0) const;\n    /// See QueryIntText()\n    uint64_t Unsigned64Text(uint64_t defaultValue = 0) const;\n\t/// See QueryIntText()\n\tbool BoolText(bool defaultValue = false) const;\n\t/// See QueryIntText()\n\tdouble DoubleText(double defaultValue = 0) const;\n\t/// See QueryIntText()\n    float FloatText(float defaultValue = 0) const;\n\n    /**\n        Convenience method to create a new XMLElement and add it as last (right)\n        child of this node. Returns the created and inserted element.\n    */\n    XMLElement* InsertNewChildElement(const char* name);\n    /// See InsertNewChildElement()\n    XMLComment* InsertNewComment(const char* comment);\n    /// See InsertNewChildElement()\n    XMLText* InsertNewText(const char* text);\n    /// See InsertNewChildElement()\n    XMLDeclaration* InsertNewDeclaration(const char* text);\n    /// See InsertNewChildElement()\n    XMLUnknown* InsertNewUnknown(const char* text);\n\n\n    // internal:\n    enum ElementClosingType {\n        OPEN,\t\t// <foo>\n        CLOSED,\t\t// <foo/>\n        CLOSING\t\t// </foo>\n    };\n    ElementClosingType ClosingType() const {\n        return _closingType;\n    }\n    virtual XMLNode* ShallowClone( XMLDocument* document ) const;\n    virtual bool ShallowEqual( const XMLNode* compare ) const;\n\nprotected:\n    char* ParseDeep( char* p, StrPair* parentEndTag, int* curLineNumPtr );\n\nprivate:\n    XMLElement( XMLDocument* doc );\n    virtual ~XMLElement();\n    XMLElement( const XMLElement& );\t// not supported\n    void operator=( const XMLElement& );\t// not supported\n\n    XMLAttribute* FindOrCreateAttribute( const char* name );\n    char* ParseAttributes( char* p, int* curLineNumPtr );\n    static void DeleteAttribute( XMLAttribute* attribute );\n    XMLAttribute* CreateAttribute();\n\n    enum { BUF_SIZE = 200 };\n    ElementClosingType _closingType;\n    // The attribute list is ordered; there is no 'lastAttribute'\n    // because the list needs to be scanned for dupes before adding\n    // a new attribute.\n    XMLAttribute* _rootAttribute;\n};\n\n\nenum Whitespace {\n    PRESERVE_WHITESPACE,\n    COLLAPSE_WHITESPACE\n};\n\n\n/** A Document binds together all the functionality.\n\tIt can be saved, loaded, and printed to the screen.\n\tAll Nodes are connected and allocated to a Document.\n\tIf the Document is deleted, all its Nodes are also deleted.\n*/\nclass TINYXML2_LIB XMLDocument : public XMLNode\n{\n    friend class XMLElement;\n    // Gives access to SetError and Push/PopDepth, but over-access for everything else.\n    // Wishing C++ had \"internal\" scope.\n    friend class XMLNode;\n    friend class XMLText;\n    friend class XMLComment;\n    friend class XMLDeclaration;\n    friend class XMLUnknown;\npublic:\n    /// constructor\n    XMLDocument( bool processEntities = true, Whitespace whitespaceMode = PRESERVE_WHITESPACE );\n    ~XMLDocument();\n\n    virtual XMLDocument* ToDocument()\t\t\t\t{\n        TIXMLASSERT( this == _document );\n        return this;\n    }\n    virtual const XMLDocument* ToDocument() const\t{\n        TIXMLASSERT( this == _document );\n        return this;\n    }\n\n    /**\n    \tParse an XML file from a character string.\n    \tReturns XML_SUCCESS (0) on success, or\n    \tan errorID.\n\n    \tYou may optionally pass in the 'nBytes', which is\n    \tthe number of bytes which will be parsed. If not\n    \tspecified, TinyXML-2 will assume 'xml' points to a\n    \tnull terminated string.\n    */\n    XMLError Parse( const char* xml, size_t nBytes=static_cast<size_t>(-1) );\n\n    /**\n    \tLoad an XML file from disk.\n    \tReturns XML_SUCCESS (0) on success, or\n    \tan errorID.\n    */\n    XMLError LoadFile( const char* filename );\n\n    /**\n    \tLoad an XML file from disk. You are responsible\n    \tfor providing and closing the FILE*.\n\n        NOTE: The file should be opened as binary (\"rb\")\n        not text in order for TinyXML-2 to correctly\n        do newline normalization.\n\n    \tReturns XML_SUCCESS (0) on success, or\n    \tan errorID.\n    */\n    XMLError LoadFile( FILE* );\n\n    /**\n    \tSave the XML file to disk.\n    \tReturns XML_SUCCESS (0) on success, or\n    \tan errorID.\n    */\n    XMLError SaveFile( const char* filename, bool compact = false );\n\n    /**\n    \tSave the XML file to disk. You are responsible\n    \tfor providing and closing the FILE*.\n\n    \tReturns XML_SUCCESS (0) on success, or\n    \tan errorID.\n    */\n    XMLError SaveFile( FILE* fp, bool compact = false );\n\n    bool ProcessEntities() const\t\t{\n        return _processEntities;\n    }\n    Whitespace WhitespaceMode() const\t{\n        return _whitespaceMode;\n    }\n\n    /**\n    \tReturns true if this document has a leading Byte Order Mark of UTF8.\n    */\n    bool HasBOM() const {\n        return _writeBOM;\n    }\n    /** Sets whether to write the BOM when writing the file.\n    */\n    void SetBOM( bool useBOM ) {\n        _writeBOM = useBOM;\n    }\n\n    /** Return the root element of DOM. Equivalent to FirstChildElement().\n        To get the first node, use FirstChild().\n    */\n    XMLElement* RootElement()\t\t\t\t{\n        return FirstChildElement();\n    }\n    const XMLElement* RootElement() const\t{\n        return FirstChildElement();\n    }\n\n    /** Print the Document. If the Printer is not provided, it will\n        print to stdout. If you provide Printer, this can print to a file:\n    \t@verbatim\n    \tXMLPrinter printer( fp );\n    \tdoc.Print( &printer );\n    \t@endverbatim\n\n    \tOr you can use a printer to print to memory:\n    \t@verbatim\n    \tXMLPrinter printer;\n    \tdoc.Print( &printer );\n    \t// printer.CStr() has a const char* to the XML\n    \t@endverbatim\n    */\n    void Print( XMLPrinter* streamer=0 ) const;\n    virtual bool Accept( XMLVisitor* visitor ) const;\n\n    /**\n    \tCreate a new Element associated with\n    \tthis Document. The memory for the Element\n    \tis managed by the Document.\n    */\n    XMLElement* NewElement( const char* name );\n    /**\n    \tCreate a new Comment associated with\n    \tthis Document. The memory for the Comment\n    \tis managed by the Document.\n    */\n    XMLComment* NewComment( const char* comment );\n    /**\n    \tCreate a new Text associated with\n    \tthis Document. The memory for the Text\n    \tis managed by the Document.\n    */\n    XMLText* NewText( const char* text );\n    /**\n    \tCreate a new Declaration associated with\n    \tthis Document. The memory for the object\n    \tis managed by the Document.\n\n    \tIf the 'text' param is null, the standard\n    \tdeclaration is used.:\n    \t@verbatim\n    \t\t<?xml version=\"1.0\" encoding=\"UTF-8\"?>\n    \t@endverbatim\n    */\n    XMLDeclaration* NewDeclaration( const char* text=0 );\n    /**\n    \tCreate a new Unknown associated with\n    \tthis Document. The memory for the object\n    \tis managed by the Document.\n    */\n    XMLUnknown* NewUnknown( const char* text );\n\n    /**\n    \tDelete a node associated with this document.\n    \tIt will be unlinked from the DOM.\n    */\n    void DeleteNode( XMLNode* node );\n\n    void ClearError() {\n        SetError(XML_SUCCESS, 0, 0);\n    }\n\n    /// Return true if there was an error parsing the document.\n    bool Error() const {\n        return _errorID != XML_SUCCESS;\n    }\n    /// Return the errorID.\n    XMLError  ErrorID() const {\n        return _errorID;\n    }\n\tconst char* ErrorName() const;\n    static const char* ErrorIDToName(XMLError errorID);\n\n    /** Returns a \"long form\" error description. A hopefully helpful\n        diagnostic with location, line number, and/or additional info.\n    */\n\tconst char* ErrorStr() const;\n\n    /// A (trivial) utility function that prints the ErrorStr() to stdout.\n    void PrintError() const;\n\n    /// Return the line where the error occurred, or zero if unknown.\n    int ErrorLineNum() const\n    {\n        return _errorLineNum;\n    }\n\n    /// Clear the document, resetting it to the initial state.\n    void Clear();\n\n\t/**\n\t\tCopies this document to a target document.\n\t\tThe target will be completely cleared before the copy.\n\t\tIf you want to copy a sub-tree, see XMLNode::DeepClone().\n\n\t\tNOTE: that the 'target' must be non-null.\n\t*/\n\tvoid DeepCopy(XMLDocument* target) const;\n\n\t// internal\n    char* Identify( char* p, XMLNode** node );\n\n\t// internal\n\tvoid MarkInUse(const XMLNode* const);\n\n    virtual XMLNode* ShallowClone( XMLDocument* /*document*/ ) const\t{\n        return 0;\n    }\n    virtual bool ShallowEqual( const XMLNode* /*compare*/ ) const\t{\n        return false;\n    }\n\nprivate:\n    XMLDocument( const XMLDocument& );\t// not supported\n    void operator=( const XMLDocument& );\t// not supported\n\n    bool\t\t\t_writeBOM;\n    bool\t\t\t_processEntities;\n    XMLError\t\t_errorID;\n    Whitespace\t\t_whitespaceMode;\n    mutable StrPair\t_errorStr;\n    int             _errorLineNum;\n    char*\t\t\t_charBuffer;\n    int\t\t\t\t_parseCurLineNum;\n\tint\t\t\t\t_parsingDepth;\n\t// Memory tracking does add some overhead.\n\t// However, the code assumes that you don't\n\t// have a bunch of unlinked nodes around.\n\t// Therefore it takes less memory to track\n\t// in the document vs. a linked list in the XMLNode,\n\t// and the performance is the same.\n\tDynArray<XMLNode*, 10> _unlinked;\n\n    MemPoolT< sizeof(XMLElement) >\t _elementPool;\n    MemPoolT< sizeof(XMLAttribute) > _attributePool;\n    MemPoolT< sizeof(XMLText) >\t\t _textPool;\n    MemPoolT< sizeof(XMLComment) >\t _commentPool;\n\n\tstatic const char* _errorNames[XML_ERROR_COUNT];\n\n    void Parse();\n\n    void SetError( XMLError error, int lineNum, const char* format, ... );\n\n\t// Something of an obvious security hole, once it was discovered.\n\t// Either an ill-formed XML or an excessively deep one can overflow\n\t// the stack. Track stack depth, and error out if needed.\n\tclass DepthTracker {\n\tpublic:\n\t\texplicit DepthTracker(XMLDocument * document) {\n\t\t\tthis->_document = document;\n\t\t\tdocument->PushDepth();\n\t\t}\n\t\t~DepthTracker() {\n\t\t\t_document->PopDepth();\n\t\t}\n\tprivate:\n\t\tXMLDocument * _document;\n\t};\n\tvoid PushDepth();\n\tvoid PopDepth();\n\n    template<class NodeType, int PoolElementSize>\n    NodeType* CreateUnlinkedNode( MemPoolT<PoolElementSize>& pool );\n};\n\ntemplate<class NodeType, int PoolElementSize>\ninline NodeType* XMLDocument::CreateUnlinkedNode( MemPoolT<PoolElementSize>& pool )\n{\n    TIXMLASSERT( sizeof( NodeType ) == PoolElementSize );\n    TIXMLASSERT( sizeof( NodeType ) == pool.ItemSize() );\n    NodeType* returnNode = new (pool.Alloc()) NodeType( this );\n    TIXMLASSERT( returnNode );\n    returnNode->_memPool = &pool;\n\n\t_unlinked.Push(returnNode);\n    return returnNode;\n}\n\n/**\n\tA XMLHandle is a class that wraps a node pointer with null checks; this is\n\tan incredibly useful thing. Note that XMLHandle is not part of the TinyXML-2\n\tDOM structure. It is a separate utility class.\n\n\tTake an example:\n\t@verbatim\n\t<Document>\n\t\t<Element attributeA = \"valueA\">\n\t\t\t<Child attributeB = \"value1\" />\n\t\t\t<Child attributeB = \"value2\" />\n\t\t</Element>\n\t</Document>\n\t@endverbatim\n\n\tAssuming you want the value of \"attributeB\" in the 2nd \"Child\" element, it's very\n\teasy to write a *lot* of code that looks like:\n\n\t@verbatim\n\tXMLElement* root = document.FirstChildElement( \"Document\" );\n\tif ( root )\n\t{\n\t\tXMLElement* element = root->FirstChildElement( \"Element\" );\n\t\tif ( element )\n\t\t{\n\t\t\tXMLElement* child = element->FirstChildElement( \"Child\" );\n\t\t\tif ( child )\n\t\t\t{\n\t\t\t\tXMLElement* child2 = child->NextSiblingElement( \"Child\" );\n\t\t\t\tif ( child2 )\n\t\t\t\t{\n\t\t\t\t\t// Finally do something useful.\n\t@endverbatim\n\n\tAnd that doesn't even cover \"else\" cases. XMLHandle addresses the verbosity\n\tof such code. A XMLHandle checks for null pointers so it is perfectly safe\n\tand correct to use:\n\n\t@verbatim\n\tXMLHandle docHandle( &document );\n\tXMLElement* child2 = docHandle.FirstChildElement( \"Document\" ).FirstChildElement( \"Element\" ).FirstChildElement().NextSiblingElement();\n\tif ( child2 )\n\t{\n\t\t// do something useful\n\t@endverbatim\n\n\tWhich is MUCH more concise and useful.\n\n\tIt is also safe to copy handles - internally they are nothing more than node pointers.\n\t@verbatim\n\tXMLHandle handleCopy = handle;\n\t@endverbatim\n\n\tSee also XMLConstHandle, which is the same as XMLHandle, but operates on const objects.\n*/\nclass TINYXML2_LIB XMLHandle\n{\npublic:\n    /// Create a handle from any node (at any depth of the tree.) This can be a null pointer.\n    explicit XMLHandle( XMLNode* node ) : _node( node ) {\n    }\n    /// Create a handle from a node.\n    explicit XMLHandle( XMLNode& node ) : _node( &node ) {\n    }\n    /// Copy constructor\n    XMLHandle( const XMLHandle& ref ) : _node( ref._node ) {\n    }\n    /// Assignment\n    XMLHandle& operator=( const XMLHandle& ref )\t\t\t\t\t\t\t{\n        _node = ref._node;\n        return *this;\n    }\n\n    /// Get the first child of this handle.\n    XMLHandle FirstChild() \t\t\t\t\t\t\t\t\t\t\t\t\t{\n        return XMLHandle( _node ? _node->FirstChild() : 0 );\n    }\n    /// Get the first child element of this handle.\n    XMLHandle FirstChildElement( const char* name = 0 )\t\t\t\t\t\t{\n        return XMLHandle( _node ? _node->FirstChildElement( name ) : 0 );\n    }\n    /// Get the last child of this handle.\n    XMLHandle LastChild()\t\t\t\t\t\t\t\t\t\t\t\t\t{\n        return XMLHandle( _node ? _node->LastChild() : 0 );\n    }\n    /// Get the last child element of this handle.\n    XMLHandle LastChildElement( const char* name = 0 )\t\t\t\t\t\t{\n        return XMLHandle( _node ? _node->LastChildElement( name ) : 0 );\n    }\n    /// Get the previous sibling of this handle.\n    XMLHandle PreviousSibling()\t\t\t\t\t\t\t\t\t\t\t\t{\n        return XMLHandle( _node ? _node->PreviousSibling() : 0 );\n    }\n    /// Get the previous sibling element of this handle.\n    XMLHandle PreviousSiblingElement( const char* name = 0 )\t\t\t\t{\n        return XMLHandle( _node ? _node->PreviousSiblingElement( name ) : 0 );\n    }\n    /// Get the next sibling of this handle.\n    XMLHandle NextSibling()\t\t\t\t\t\t\t\t\t\t\t\t\t{\n        return XMLHandle( _node ? _node->NextSibling() : 0 );\n    }\n    /// Get the next sibling element of this handle.\n    XMLHandle NextSiblingElement( const char* name = 0 )\t\t\t\t\t{\n        return XMLHandle( _node ? _node->NextSiblingElement( name ) : 0 );\n    }\n\n    /// Safe cast to XMLNode. This can return null.\n    XMLNode* ToNode()\t\t\t\t\t\t\t{\n        return _node;\n    }\n    /// Safe cast to XMLElement. This can return null.\n    XMLElement* ToElement() \t\t\t\t\t{\n        return ( _node ? _node->ToElement() : 0 );\n    }\n    /// Safe cast to XMLText. This can return null.\n    XMLText* ToText() \t\t\t\t\t\t\t{\n        return ( _node ? _node->ToText() : 0 );\n    }\n    /// Safe cast to XMLUnknown. This can return null.\n    XMLUnknown* ToUnknown() \t\t\t\t\t{\n        return ( _node ? _node->ToUnknown() : 0 );\n    }\n    /// Safe cast to XMLDeclaration. This can return null.\n    XMLDeclaration* ToDeclaration() \t\t\t{\n        return ( _node ? _node->ToDeclaration() : 0 );\n    }\n\nprivate:\n    XMLNode* _node;\n};\n\n\n/**\n\tA variant of the XMLHandle class for working with const XMLNodes and Documents. It is the\n\tsame in all regards, except for the 'const' qualifiers. See XMLHandle for API.\n*/\nclass TINYXML2_LIB XMLConstHandle\n{\npublic:\n    explicit XMLConstHandle( const XMLNode* node ) : _node( node ) {\n    }\n    explicit XMLConstHandle( const XMLNode& node ) : _node( &node ) {\n    }\n    XMLConstHandle( const XMLConstHandle& ref ) : _node( ref._node ) {\n    }\n\n    XMLConstHandle& operator=( const XMLConstHandle& ref )\t\t\t\t\t\t\t{\n        _node = ref._node;\n        return *this;\n    }\n\n    const XMLConstHandle FirstChild() const\t\t\t\t\t\t\t\t\t\t\t{\n        return XMLConstHandle( _node ? _node->FirstChild() : 0 );\n    }\n    const XMLConstHandle FirstChildElement( const char* name = 0 ) const\t\t\t\t{\n        return XMLConstHandle( _node ? _node->FirstChildElement( name ) : 0 );\n    }\n    const XMLConstHandle LastChild()\tconst\t\t\t\t\t\t\t\t\t\t{\n        return XMLConstHandle( _node ? _node->LastChild() : 0 );\n    }\n    const XMLConstHandle LastChildElement( const char* name = 0 ) const\t\t\t\t{\n        return XMLConstHandle( _node ? _node->LastChildElement( name ) : 0 );\n    }\n    const XMLConstHandle PreviousSibling() const\t\t\t\t\t\t\t\t\t{\n        return XMLConstHandle( _node ? _node->PreviousSibling() : 0 );\n    }\n    const XMLConstHandle PreviousSiblingElement( const char* name = 0 ) const\t\t{\n        return XMLConstHandle( _node ? _node->PreviousSiblingElement( name ) : 0 );\n    }\n    const XMLConstHandle NextSibling() const\t\t\t\t\t\t\t\t\t\t{\n        return XMLConstHandle( _node ? _node->NextSibling() : 0 );\n    }\n    const XMLConstHandle NextSiblingElement( const char* name = 0 ) const\t\t\t{\n        return XMLConstHandle( _node ? _node->NextSiblingElement( name ) : 0 );\n    }\n\n\n    const XMLNode* ToNode() const\t\t\t\t{\n        return _node;\n    }\n    const XMLElement* ToElement() const\t\t\t{\n        return ( _node ? _node->ToElement() : 0 );\n    }\n    const XMLText* ToText() const\t\t\t\t{\n        return ( _node ? _node->ToText() : 0 );\n    }\n    const XMLUnknown* ToUnknown() const\t\t\t{\n        return ( _node ? _node->ToUnknown() : 0 );\n    }\n    const XMLDeclaration* ToDeclaration() const\t{\n        return ( _node ? _node->ToDeclaration() : 0 );\n    }\n\nprivate:\n    const XMLNode* _node;\n};\n\n\n/**\n\tPrinting functionality. The XMLPrinter gives you more\n\toptions than the XMLDocument::Print() method.\n\n\tIt can:\n\t-# Print to memory.\n\t-# Print to a file you provide.\n\t-# Print XML without a XMLDocument.\n\n\tPrint to Memory\n\n\t@verbatim\n\tXMLPrinter printer;\n\tdoc.Print( &printer );\n\tSomeFunction( printer.CStr() );\n\t@endverbatim\n\n\tPrint to a File\n\n\tYou provide the file pointer.\n\t@verbatim\n\tXMLPrinter printer( fp );\n\tdoc.Print( &printer );\n\t@endverbatim\n\n\tPrint without a XMLDocument\n\n\tWhen loading, an XML parser is very useful. However, sometimes\n\twhen saving, it just gets in the way. The code is often set up\n\tfor streaming, and constructing the DOM is just overhead.\n\n\tThe Printer supports the streaming case. The following code\n\tprints out a trivially simple XML file without ever creating\n\tan XML document.\n\n\t@verbatim\n\tXMLPrinter printer( fp );\n\tprinter.OpenElement( \"foo\" );\n\tprinter.PushAttribute( \"foo\", \"bar\" );\n\tprinter.CloseElement();\n\t@endverbatim\n*/\nclass TINYXML2_LIB XMLPrinter : public XMLVisitor\n{\npublic:\n    /** Construct the printer. If the FILE* is specified,\n    \tthis will print to the FILE. Else it will print\n    \tto memory, and the result is available in CStr().\n    \tIf 'compact' is set to true, then output is created\n    \twith only required whitespace and newlines.\n    */\n    XMLPrinter( FILE* file=0, bool compact = false, int depth = 0 );\n    virtual ~XMLPrinter()\t{}\n\n    /** If streaming, write the BOM and declaration. */\n    void PushHeader( bool writeBOM, bool writeDeclaration );\n    /** If streaming, start writing an element.\n        The element must be closed with CloseElement()\n    */\n    void OpenElement( const char* name, bool compactMode=false );\n    /// If streaming, add an attribute to an open element.\n    void PushAttribute( const char* name, const char* value );\n    void PushAttribute( const char* name, int value );\n    void PushAttribute( const char* name, unsigned value );\n\tvoid PushAttribute( const char* name, int64_t value );\n\tvoid PushAttribute( const char* name, uint64_t value );\n\tvoid PushAttribute( const char* name, bool value );\n    void PushAttribute( const char* name, double value );\n    /// If streaming, close the Element.\n    virtual void CloseElement( bool compactMode=false );\n\n    /// Add a text node.\n    void PushText( const char* text, bool cdata=false );\n    /// Add a text node from an integer.\n    void PushText( int value );\n    /// Add a text node from an unsigned.\n    void PushText( unsigned value );\n\t/// Add a text node from a signed 64bit integer.\n\tvoid PushText( int64_t value );\n\t/// Add a text node from an unsigned 64bit integer.\n\tvoid PushText( uint64_t value );\n\t/// Add a text node from a bool.\n    void PushText( bool value );\n    /// Add a text node from a float.\n    void PushText( float value );\n    /// Add a text node from a double.\n    void PushText( double value );\n\n    /// Add a comment\n    void PushComment( const char* comment );\n\n    void PushDeclaration( const char* value );\n    void PushUnknown( const char* value );\n\n    virtual bool VisitEnter( const XMLDocument& /*doc*/ );\n    virtual bool VisitExit( const XMLDocument& /*doc*/ )\t\t\t{\n        return true;\n    }\n\n    virtual bool VisitEnter( const XMLElement& element, const XMLAttribute* attribute );\n    virtual bool VisitExit( const XMLElement& element );\n\n    virtual bool Visit( const XMLText& text );\n    virtual bool Visit( const XMLComment& comment );\n    virtual bool Visit( const XMLDeclaration& declaration );\n    virtual bool Visit( const XMLUnknown& unknown );\n\n    /**\n    \tIf in print to memory mode, return a pointer to\n    \tthe XML file in memory.\n    */\n    const char* CStr() const {\n        return _buffer.Mem();\n    }\n    /**\n    \tIf in print to memory mode, return the size\n    \tof the XML file in memory. (Note the size returned\n    \tincludes the terminating null.)\n    */\n    int CStrSize() const {\n        return _buffer.Size();\n    }\n    /**\n    \tIf in print to memory mode, reset the buffer to the\n    \tbeginning.\n    */\n    void ClearBuffer( bool resetToFirstElement = true ) {\n        _buffer.Clear();\n        _buffer.Push(0);\n\t\t_firstElement = resetToFirstElement;\n    }\n\nprotected:\n\tvirtual bool CompactMode( const XMLElement& )\t{ return _compactMode; }\n\n\t/** Prints out the space before an element. You may override to change\n\t    the space and tabs used. A PrintSpace() override should call Print().\n\t*/\n    virtual void PrintSpace( int depth );\n    virtual void Print( const char* format, ... );\n    virtual void Write( const char* data, size_t size );\n    virtual void Putc( char ch );\n\n    inline void Write(const char* data) { Write(data, strlen(data)); }\n\n    void SealElementIfJustOpened();\n    bool _elementJustOpened;\n    DynArray< const char*, 10 > _stack;\n\nprivate:\n    /**\n       Prepares to write a new node. This includes sealing an element that was\n       just opened, and writing any whitespace necessary if not in compact mode.\n     */\n    void PrepareForNewNode( bool compactMode );\n    void PrintString( const char*, bool restrictedEntitySet );\t// prints out, after detecting entities.\n\n    bool _firstElement;\n    FILE* _fp;\n    int _depth;\n    int _textDepth;\n    bool _processEntities;\n\tbool _compactMode;\n\n    enum {\n        ENTITY_RANGE = 64,\n        BUF_SIZE = 200\n    };\n    bool _entityFlag[ENTITY_RANGE];\n    bool _restrictedEntityFlag[ENTITY_RANGE];\n\n    DynArray< char, 20 > _buffer;\n\n    // Prohibit cloning, intentionally not implemented\n    XMLPrinter( const XMLPrinter& );\n    XMLPrinter& operator=( const XMLPrinter& );\n};\n\n\n}\t// tinyxml2\n\n#if defined(_MSC_VER)\n#   pragma warning(pop)\n#endif\n\n#endif // TINYXML2_INCLUDED\n"
  },
  {
    "path": "TrafficMonitor.sln",
    "content": "﻿\nMicrosoft Visual Studio Solution File, Format Version 12.00\n# Visual Studio Version 16\nVisualStudioVersion = 16.0.31515.178\nMinimumVisualStudioVersion = 10.0.40219.1\nProject(\"{8BC9CEB8-8B4A-11D0-8D11-00A0C91BC942}\") = \"TrafficMonitor\", \"TrafficMonitor\\TrafficMonitor.vcxproj\", \"{09483BED-B1E9-4827-8120-A18302C84AA8}\"\n\tProjectSection(ProjectDependencies) = postProject\n\t\t{C0A42F4A-ABB3-4575-B4D5-CEDD8379AC26} = {C0A42F4A-ABB3-4575-B4D5-CEDD8379AC26}\n\tEndProjectSection\nEndProject\nProject(\"{8BC9CEB8-8B4A-11D0-8D11-00A0C91BC942}\") = \"OpenHardwareMonitorApi\", \"OpenHardwareMonitorApi\\OpenHardwareMonitorApi.vcxproj\", \"{C0A42F4A-ABB3-4575-B4D5-CEDD8379AC26}\"\nEndProject\nProject(\"{8BC9CEB8-8B4A-11D0-8D11-00A0C91BC942}\") = \"PluginDemo\", \"PluginDemo\\PluginDemo.vcxproj\", \"{D1CA3ECC-DC32-445A-B734-C4DB08D4BA34}\"\nEndProject\nGlobal\n\tGlobalSection(SolutionConfigurationPlatforms) = preSolution\n\t\tDebug (lite)|x64 = Debug (lite)|x64\n\t\tDebug (lite)|x86 = Debug (lite)|x86\n\t\tDebug|x64 = Debug|x64\n\t\tDebug|x86 = Debug|x86\n\t\tRelease (lite)|x64 = Release (lite)|x64\n\t\tRelease (lite)|x86 = Release (lite)|x86\n\t\tRelease|x64 = Release|x64\n\t\tRelease|x86 = Release|x86\n\tEndGlobalSection\n\tGlobalSection(ProjectConfigurationPlatforms) = postSolution\n\t\t{09483BED-B1E9-4827-8120-A18302C84AA8}.Debug (lite)|x64.ActiveCfg = Debug (lite)|x64\n\t\t{09483BED-B1E9-4827-8120-A18302C84AA8}.Debug (lite)|x64.Build.0 = Debug (lite)|x64\n\t\t{09483BED-B1E9-4827-8120-A18302C84AA8}.Debug (lite)|x86.ActiveCfg = Debug (lite)|Win32\n\t\t{09483BED-B1E9-4827-8120-A18302C84AA8}.Debug (lite)|x86.Build.0 = Debug (lite)|Win32\n\t\t{09483BED-B1E9-4827-8120-A18302C84AA8}.Debug|x64.ActiveCfg = Debug|x64\n\t\t{09483BED-B1E9-4827-8120-A18302C84AA8}.Debug|x64.Build.0 = Debug|x64\n\t\t{09483BED-B1E9-4827-8120-A18302C84AA8}.Debug|x86.ActiveCfg = Debug|Win32\n\t\t{09483BED-B1E9-4827-8120-A18302C84AA8}.Debug|x86.Build.0 = Debug|Win32\n\t\t{09483BED-B1E9-4827-8120-A18302C84AA8}.Release (lite)|x64.ActiveCfg = Release (lite)|x64\n\t\t{09483BED-B1E9-4827-8120-A18302C84AA8}.Release (lite)|x64.Build.0 = Release (lite)|x64\n\t\t{09483BED-B1E9-4827-8120-A18302C84AA8}.Release (lite)|x86.ActiveCfg = Release (lite)|Win32\n\t\t{09483BED-B1E9-4827-8120-A18302C84AA8}.Release (lite)|x86.Build.0 = Release (lite)|Win32\n\t\t{09483BED-B1E9-4827-8120-A18302C84AA8}.Release|x64.ActiveCfg = Release|x64\n\t\t{09483BED-B1E9-4827-8120-A18302C84AA8}.Release|x64.Build.0 = Release|x64\n\t\t{09483BED-B1E9-4827-8120-A18302C84AA8}.Release|x86.ActiveCfg = Release|Win32\n\t\t{09483BED-B1E9-4827-8120-A18302C84AA8}.Release|x86.Build.0 = Release|Win32\n\t\t{C0A42F4A-ABB3-4575-B4D5-CEDD8379AC26}.Debug (lite)|x64.ActiveCfg = Debug|x64\n\t\t{C0A42F4A-ABB3-4575-B4D5-CEDD8379AC26}.Debug (lite)|x86.ActiveCfg = Debug|Win32\n\t\t{C0A42F4A-ABB3-4575-B4D5-CEDD8379AC26}.Debug|x64.ActiveCfg = Debug|x64\n\t\t{C0A42F4A-ABB3-4575-B4D5-CEDD8379AC26}.Debug|x64.Build.0 = Debug|x64\n\t\t{C0A42F4A-ABB3-4575-B4D5-CEDD8379AC26}.Debug|x86.ActiveCfg = Debug|Win32\n\t\t{C0A42F4A-ABB3-4575-B4D5-CEDD8379AC26}.Debug|x86.Build.0 = Debug|Win32\n\t\t{C0A42F4A-ABB3-4575-B4D5-CEDD8379AC26}.Release (lite)|x64.ActiveCfg = Release|x64\n\t\t{C0A42F4A-ABB3-4575-B4D5-CEDD8379AC26}.Release (lite)|x86.ActiveCfg = Release|Win32\n\t\t{C0A42F4A-ABB3-4575-B4D5-CEDD8379AC26}.Release|x64.ActiveCfg = Release|x64\n\t\t{C0A42F4A-ABB3-4575-B4D5-CEDD8379AC26}.Release|x64.Build.0 = Release|x64\n\t\t{C0A42F4A-ABB3-4575-B4D5-CEDD8379AC26}.Release|x86.ActiveCfg = Release|Win32\n\t\t{C0A42F4A-ABB3-4575-B4D5-CEDD8379AC26}.Release|x86.Build.0 = Release|Win32\n\t\t{D1CA3ECC-DC32-445A-B734-C4DB08D4BA34}.Debug (lite)|x64.ActiveCfg = Debug|x64\n\t\t{D1CA3ECC-DC32-445A-B734-C4DB08D4BA34}.Debug (lite)|x64.Build.0 = Debug|x64\n\t\t{D1CA3ECC-DC32-445A-B734-C4DB08D4BA34}.Debug (lite)|x86.ActiveCfg = Debug|Win32\n\t\t{D1CA3ECC-DC32-445A-B734-C4DB08D4BA34}.Debug (lite)|x86.Build.0 = Debug|Win32\n\t\t{D1CA3ECC-DC32-445A-B734-C4DB08D4BA34}.Debug|x64.ActiveCfg = Debug|x64\n\t\t{D1CA3ECC-DC32-445A-B734-C4DB08D4BA34}.Debug|x64.Build.0 = Debug|x64\n\t\t{D1CA3ECC-DC32-445A-B734-C4DB08D4BA34}.Debug|x86.ActiveCfg = Debug|Win32\n\t\t{D1CA3ECC-DC32-445A-B734-C4DB08D4BA34}.Debug|x86.Build.0 = Debug|Win32\n\t\t{D1CA3ECC-DC32-445A-B734-C4DB08D4BA34}.Release (lite)|x64.ActiveCfg = Release|x64\n\t\t{D1CA3ECC-DC32-445A-B734-C4DB08D4BA34}.Release (lite)|x64.Build.0 = Release|x64\n\t\t{D1CA3ECC-DC32-445A-B734-C4DB08D4BA34}.Release (lite)|x86.ActiveCfg = Release|Win32\n\t\t{D1CA3ECC-DC32-445A-B734-C4DB08D4BA34}.Release (lite)|x86.Build.0 = Release|Win32\n\t\t{D1CA3ECC-DC32-445A-B734-C4DB08D4BA34}.Release|x64.ActiveCfg = Release|x64\n\t\t{D1CA3ECC-DC32-445A-B734-C4DB08D4BA34}.Release|x64.Build.0 = Release|x64\n\t\t{D1CA3ECC-DC32-445A-B734-C4DB08D4BA34}.Release|x86.ActiveCfg = Release|Win32\n\t\t{D1CA3ECC-DC32-445A-B734-C4DB08D4BA34}.Release|x86.Build.0 = Release|Win32\n\tEndGlobalSection\n\tGlobalSection(SolutionProperties) = preSolution\n\t\tHideSolutionNode = FALSE\n\tEndGlobalSection\n\tGlobalSection(ExtensibilityGlobals) = postSolution\n\t\tSolutionGuid = {2DE8C8C3-89CE-4313-978B-DADEC5CB141A}\n\tEndGlobalSection\nEndGlobal\n"
  },
  {
    "path": "UpdateLog/update_log.md",
    "content": "**简体中文 | [繁體中文](./update_log_zh-tw.md) | [English](./update_log_en-us.md)**\n\n# TrafficMonitor 更新日志\n## V1.83 (2022/02/03)\n\n* 修正当已经存在一个开机自启动的任务计划时，无法设置开机自启动的问题，\n* 新增当已经存在TrafficMonitor进程时启动程序，弹出“程序已经在运行”对话框。\n* 鼠标提示中新增插件信息的显示。\n* 修正插件被禁用后仍然会被加载的问题。\n* 任务栏中的插件项目支持自由排序。\n* 选项设置中新增可以选择要在“网络连接列表”中显示的网络的功能。\n* 新增显示总网速的功能。\n* 新增在任务栏中显示网速占用图的功能。\n* 修正硬件监控中有多个名称相同的硬盘时只能显示一个硬盘的问题。\n* 优化任务栏窗口横向滚动图的显示效果。\n* 任务栏窗口右键菜单中的“显示设置”由菜单改为对话框形式。\n* “选择网络连接”菜单中添加“刷新网络连接列表”命令。\n* 修正几处导致程序崩溃的问题。\n\n## V1.82 (2021/12/12)\n\n* 新增插件系统，可以通过插件在任务栏窗口和主窗口显示更多自定义内容。\n* 任务栏右键菜单中增加“任务管理器”命令。\n* 选项设置中新增“应用”按钮。\n* 历史流量统计中增加周视图。\n* 新增任务栏项目间距的设置。\n* 修正Windows11深色模式下任务栏右键菜单无法弹出的问题。\n* 修正Windows11下使用StartAllBack等软件将任务栏恢复为Windows10样式时，任务栏窗口无法显示的问题。\n\n## V1.81 (2021/07/27)\n\n* 修正1.80版本以来的一些崩溃的问题\n* 任务栏窗口显示项目支持自定义排序\n* 任务栏窗口中显示的项目数量为奇数时，最后一个项目垂直排列，以节省任务栏空间\n* 新增显示硬盘利用率的功能\n* 将右键菜单中的一些设置添加到选项设置界面中\n* 新增主窗口对多显示器的支持，在不勾选“允许超出屏幕边界”的情况下也能移动到其他显示器了\n\n## V1.80.3 (2021/05/29)\n\n* 修正无法显示CPU温度的问题\n* 新增可以选择监控指定CPU核心的温度的功能\n* 修正一处导致程序崩溃的问题\n\n## V1.80.2 (2021/05/22)\n\n* 修正几处导致程序崩溃的问题\n* 获取温度信息改用LibreHardwareMonitor\n* 新增可选择要监控哪块硬盘的温度的功能\n* 新增可选择要监控哪些硬件的功能\n* 修正程序退出时LibreHardwareMonitorLib.sys文件没有解除占用的问题\n\n## V1.80.1 (2021/05/17)\n\n* 修正无法获取AMD显卡温度的问题\n* 修正自动切换为浅色模式颜色预设时程序崩溃的问题\n* 修正无法获取11代Intel处理器温度的问题\n* 修正设置开机自启动无效的问题\n\n## V1.80 (2021/05/15)\n\n* 新增CPU、显卡、硬盘和主板温度监控的功能\n* 新增显卡利用率监控的功能\n* 修正历史流量统计数值溢出导致统计数值不正确的问题\n* 修正屏幕分辨率更改后主窗口位置不正确的问题\n* 修正系统DPI更改后，主窗口和任务栏窗口界面元素的大小不会随DPI变化的问题\n* 新增任务栏窗口自由指定显示项目的功能\n* 新增xml格式的主窗口皮肤，支持温度显示\n* 修正使用触屏设备时，选项设置中的子窗口无法使用触屏滚动的问题\n* 将检查更新的处理放到后台线程中，以解决检查更新时程序长时间没有响应的问题\n* 修正任务栏窗口中按Alt+F4后程序异常的问题\n* 新增对每秒4GB以上网速的支持\n* 新增可选择更新源为Gitee，以解决中国大陆地区有时无法访问GitHub导致程序无法更新的问题\n* 新增内存显示方式设置\n* 修正当前监控的网卡被关闭或禁用，再次启动后无法自动选择之前监控的网卡的问题\n* 开机自启动采用任务计划实现，以解决有时开机自启动无效的问题\n* 修正一些崩溃的问题\n* 其他细节方面的改进\n\n## V1.79.1 (2020/08/05)\n\n* 修正注册表句柄泄漏的问题。\n* 修正当主窗口和任务栏窗口都不显示时，不统计CPU和内存利用率的问题。\n\n## V1.79 (2020/07/30)\n\n* 新增任务栏窗口颜色自动适应Windows10深色/浅色主题的功能。\n* 新增通知区图标自动适应Windows10深色/浅色主题的功能。\n* 增加CPU获取方式的设置，解决部分用户出现的CPU利用率始终为0的问题。\n* 选项设置>任务栏设置中去掉“透明色”的设置，新增“背景色透明”的选项。\n* 新增允许任务栏只显示CPU和内存利用率而不显示网速。\n* 修正了关机后设置可能会没有保存的问题。\n* 历史流量统计-日历视图中增加了每周第一天的设置。\n* 历史流量统计-列表视图中增加了按年、月、日、季度统计的功能。\n* 可能解决了历史流量数据有小概率丢失的问题。\n* 修正Windows10浅色主题时，如果任务栏窗口背景设置为透明，会无法弹出右键菜单的问题。\n* 修正流量统计不支持统计超过2TB数据的问题。\n* 为菜单项添加了图标。\n* 当程序所在目录无法写入时，将数据保存到AppData目录。\n* 新增设置监控时间间隔的功能。\n\n## V1.78 (2020/03/21)\n\n* 新增双击主窗口或任务栏窗口打开指定应用程序的功能\n* 新增任务栏窗口中显示CPU和内存利用率状态条的功能\n* 修正Windows10中CPU利用率和任务管理器不一致的问题\n* 新增是否显示鼠标提示的选项\n* 新增程序首次启启动时根据Windows10浅色模式设置任务栏颜色的功能\n* 任务栏设置中增加预设方案的功能\n* 其他细节方面的改进\n\n## V1.77 (2019/05/01)\n* 增加任务窗口透明色设置，修正任务栏为白色时任务栏窗口文字颜色无法设置为黑色的问题（在“选项”——“任务栏窗口设置”中设置透明色）\n* 新增程序崩溃时显示崩溃信息的功能\n* 修正显示任务栏窗口的情况下，资源管理器重启时会导致屏幕闪烁的问题\n* 新增鼠标指向通知区图标时显示监控信息\n* 修正使用蓝牙网络时无法显示网速的问题\n* 新增x64的版本\n* 其他细节方面的改进\n### 更新说明：\n本次更新在一定程度上解决了Win10最新版本中白色任务栏时文字看不清的问题。需要手动在“选项”——“任务栏窗口设置”——“透明色”设置透明色。当透明色不为黑色且和背景色不同时为不透明效果，如果和背景色相同则为透明效果。在Win10白色任务栏中建议如下图所示设置：\n![白色任务栏截图](https://user-images.githubusercontent.com/30562462/57004858-36b55300-6c05-11e9-89d8-9911dc99f09c.PNG)\n## V1.76 (2018/11/11)\n* 修正了分辨率更改可能会导致程序崩溃的问题；\n* 新增配置文件保存位置的选项；\n* 修正了当分辨率变化时任务栏窗口的垂直位置不正确的问题；\n* 新增今日上传和下载流量的显示，历史流量统计中增加上传和下载流量的统计；\n* 其他细节方面的改进。\n## V1.75 (2018/08/11)\n* 任务栏窗口右键菜单增加“选择网络连接”菜单项\n* 写入配置文件时改为使用UTF8编码\n* 新增网速单位为B(字节)或b(比特)的设置\n* 修正部分电脑中无法获取网速的问题\n* 修正部分电脑中会反复产生错误日志的问题\n* 皮肤11 UI调整，修正有时文本显示不全的问题\n其他细节方面的改进\n## V1.74 (2018/06/17)\n* 修正两处导致程序崩溃的问题\n* 新增：监控所有连接的网速的功能\n* 右键菜单“选择网络连接”中显示所有的网络接口，不显示loopback接口\n* 新增：任务栏窗口项目可以水平排列\n* 新增：繁体中文语言支持\n* 其他细节方面的改进。\n## V1.73 (2018/05/13)\n* 新增：多语言支持，增加英语\n* 新增：字体设置支持粗体、斜体、下划线等字体样式\n* 新增：历史流量统计中的日历视图\n* 修正：当悬浮窗在右下角时，有时程序启动后悬浮窗会往左或往上移动一段距离的问题\n* 配置文件转移到C:\\Users\\用户名\\AppData\\Roaming\\TrafficMonitor目录下，防止由于程序所在目录无法写入数据导致配置文件无法保存的问题\n* 新增：网速数据位数设置的选项；修正有时当网速超过10M/s时数据显示的全的问题\n## V1.72 (2018/04/21)\n* 新增：任务栏窗口数值右对齐的选项\n* 新增：每个项目文本颜色单独设置的功能\n* 新增：设置鼠标双击动作的功能\n* 新增：设置当流量或内存使用率超过一定值时弹出通知的功能\n* 新增：获取外网IP地址的功能\n* 其他细节方面的改进。\n## V1.71 (2018/04/06)\n* 优化背景图片的显示效果，解决了由于图片缩放时导致的失真的问题\n* 增加不规则形状皮肤的支持\n* 增加可以通过皮肤文件设定项目的对齐方式和字体\n* 改善了优化通知区图标在125%缩放比时的显示效果\n* 一些细节方面的改进。\n## V1.70 (2018/03/30)\n* 新增：自定义皮肤功能，可以通过皮肤文件更加自由地定义悬浮窗中的每个项目的大小和位置\n* “更换皮肤”对话框中预览图可以预览悬浮窗中显示的文本和字体，当预览图过大时会显示滚动条\n* “更换皮肤”对话框中增加“皮肤制作教程”和“更多皮肤下载”链接\n* 新增允许悬浮窗超出屏幕边界的选项，解决多显示器时无法移动到其他显示器的问题\n## V1.69 (2018/03/17)\n* 新增单位设置，允许隐藏单位\n* 新增：单位设置、网速显示简洁模式可应用到主窗口\n* 新增更换通知区图标的功能\n* 一些细节方面的改进\n* 增加两款皮肤。\n## V1.68 (2018/03/03)\n* 修正当程序启动时，如果设置了隐藏主窗口，主窗口还是会显示1秒钟再消失的问题\n* 修正当窗口为左上角(0,0)时，下次启动时无法记住该位置的问题\n* 修正有时在进入选项设置再点击“确定”时，会弹出“注册表项删除失败”提示框的问题\n* 新增任务栏窗口网速显示简洁模式（选项——任务栏窗口设置——网速显示简洁模式）\n## V1.67 (2018/01/21)\n* 修正当主窗口和任务栏窗口都不显示CPU和内存利用率时，鼠标提示中的CPU和内存利用率不刷新的问题\n* 修正了当任务栏左侧有自定义工具栏或快速启动栏时，任务栏窗口位置不正确的问题\n* 新增自动检查更新功能\n* 开机自启动功能改为写入注册表实现，去掉右键菜单中的“开机自动运行”项，移至选项设置——常规设置\n* 增加任务栏窗口可以放在任务栏的左侧\n* 其他细节方面的改进\n## V1.66 (2017/12/31)\n* 新增鼠标悬浮提示会实时更新\n* 优化关于启动时弹出“已经有一个程序正在运行”对话框的处理\n* 增加无法嵌入任务栏时自动重试的处理\n* 增加任务栏窗口的刷新频率，在一定程度上解决通知区域图标变化时任务栏窗口闪烁的问题\n* 历史流量统计对话框中增加图形指示\n## V1.65 (2017/11.16)\n* 修正了更新Win10秋季创意者更新后，设置了开机启动时会弹出“已经有一个程序正在运行”的对话框的问题\n* 增加了选项设置对话框，将字体和字体颜色设置放到了选项设置对话框中，增加了显示文本的设置\n* 修正了有时开机启动时会弹出“窗口无法嵌入任务栏”的提示的问题\n* 增加程序全屏时隐藏悬浮窗的选项，并修正了全屏视频或游戏时悬浮窗会自己跳出来的问题\n* 任务栏窗口的宽度不再固定，而是会根据文本宽度自动调整\n* 连接详情对话框中增加右键菜单复制功能\n## V1.64 (2017/07/28)\n* 任务栏窗口信息显示改成了双缓冲绘图，彻底修正了任务栏窗口显示出现难看的色块的问题\n* 当任务栏从桌面底部移动到两旁时，任务栏窗口会实时切换横排和竖排显示\n## V1.63 (2017/07/05)\n* 增加当无法保存设置时的警告信息\n* 优化通知区图标行为，当设置了隐藏主窗口时，双击通知区图标才显示主窗口\n* 调整了任务栏窗口显示网速部分的宽度，以修正网速过大时显示不全的问题\n* 任务栏窗口右键菜单中增加显示通知区图标选项\n* 修正有时设置了去掉通知区图标后再次启动时图标还在的问题\n* 如果设置了主窗口总是置顶，每隔5分钟自动执行一次置顶操作，以解决有时窗口没有置顶的问题\n## V1.62 (2017/05/29)\n* 启动时如果检测到历史流量统计中有重复的日期就将它们合并\n* 去掉一些菜单项中不必要的省略号\n* 其他细节方面的改进\n## V1.61 (2017/05/19)\n* 连接详情中增加物理地址、IP地址、子网掩码和默认网关几个项目显示\n* 鼠标提示信息合并为一个提示，如果设置为不显示CPU和内存利用率时在提示信息里显示\n* 修正了连接详情窗口中的鼠标提示显示在窗口后面的问题\n* 修正了主窗口没有置顶时任务栏窗口的鼠标提示在窗口后面的问题\n* 修正了切换网络连接后的1秒网速显示为一个很大的值的问题\n* 修正了当切换网络连接时统计的流量不正确的问题\n## V1.60 (2017/05/11)\n* 连接详情界面改为列表显示，增加程序已运行时间项目\n* 新增历史流量统计功能，可以记录每一天总共使用的流量\n* 关于对话框中增加捐赠功能\n* 关于对话框中增加检查更新链接，点击可跳转到百度网盘链接\n* 优化了自动选择的处理，在较大程度上解决了有时无法显示网速的问题\n* 较大程度上解决了少数情况下设置总是置顶、鼠标穿透时无效的问题\n* 新增当鼠标指向主窗口或任务栏窗口对应的项目时，提示今日已使用流量和内存使用详情\n* 更换了程序图标和通知区域图标，风格更加简洁\n## V1.54 (2017/04/29)\n* 修正当设置了隐藏主窗口和显示任务栏窗口时，在任务视图中有一个看不见的窗口的问题\n* 修正某些情况下会导致变成灰色的菜单项变回可用状态的问题\n* 修正在“自动选择”模式时，当断开WIFI再重新连接后可能会出现无法自动选择有网络的连接的问题\n 优化：当显示任务栏窗口时，允许隐藏通知区图标。\n## V1.53 (2017/03/07)\n* 修正选择皮肤时可能会出现顺序错乱的问题\n* 优化：ini文件保存选择的皮肤的名称而不是序号，以解决添加或删除皮肤后选择的皮肤不正确的问题\n* 修正将焦点设置到任务栏窗口后按回车或ESC键任务栏窗口关闭的问题\n* 新增设置字体功能\n## V1.52 (2017/03/04)\n* 修正当显示任务栏窗口时，打开或关闭通知区图标会导致资源管理器卡住的问题\n* 优化：打开任务栏窗口时，窗口中的信息可以立即显示出来\n* 新增：允许取消开机自启动，“开机自动运行”菜单项增加复选标志\n## V1.51 (2017/03/02)\n* 修正重启资源管理器之后任务栏窗口无法正常显示的问题\n* 修正重启资源管理器之后通知区图标消失的问题\n* 修正任务栏窗口上网速显示字符串过长时字符显示不全的问题\n* 修正在特定情况下可能会导致程序停止工作的问题\n* 优化：当任务栏在屏幕左侧或右侧时，任务栏窗口的4个项目以竖排显示（必须关闭任务栏窗口再打开才能生效）\n* “关于”对话框中增加“联系作者”超链接\n## V1.50 (2017/02/25)\n* 新增更换皮肤功能（点击右键菜单——其他功能——更换皮肤，新增了7套皮肤）\n* 新增当网络变化时，自动更新“选择网络连接”中的子菜单项\n* 新增允许交换上传和下载的位置\n## V1.43 (2017/02/22)\n* 优化自动选择连接的处理，不选择已断开的连接\n* 修正任务栏窗口切换“显示CPU和内容利用率”后，“隐藏主窗口”的复选框消失的问题\n* 优化：当任务栏窗口无法嵌入任务栏时（如被安全软件阻止），如果任务栏把窗口覆盖，则窗口自动置顶。\n## V1.42 (2017/02/19)\n* 优化：任务栏窗口的位置不再总是固定，当任务栏最小化区域宽度变化时自动调整任务栏窗口的位置\n* 修正了当把任务放在桌面的左边或右边时任务栏窗口位置不正确的问题\n## V1.41 (2017/02/18)\n* 新增隐藏主界面功能\n* 更改设置时实时保存，解决关机时无法保存设置的问题\n* 新增开机自动运行功能，程序会创建快捷方式到开始菜单的启动目录\n* 新增在无法嵌入任务栏时强行将窗口移动到正确的位置，并弹出提示信息\n## V1.40 (2017/02/16)\n* 增加任务栏窗口，作为工具栏样式嵌入任务栏，不再占用屏幕区域\n* 增加主悬浮窗文字颜色设置\n* 任务栏窗口支持背景颜色和文字颜色设置\n## V1.30 (2017/02/13)\n* 增加CPU和内存利用率显示。\n* 增加鼠标穿透功能，悬浮窗不再影响鼠标操作\n## V1.20 (2017/02/11)\n* 去掉任务栏图标，增加通知区图标，右击通知区图标可弹出右键菜单\n* 右键菜单增加“锁定窗口位置”、“显示通知区域图标”项目\n* 增加移动时不允许超过屏幕边界的处理\n* 增加程序只允许一个实例运行的处理\n* 连接详情中增加自程序启动以来已发送和接收字节数显示，并增加数据以KB、MB或GB为单位的显示\n* 修正了当上传速度的字符串太长时可能会导致下载速度显示不全的BUG\n## V1.00 (2017/02/10)\n* 初版发行\n"
  },
  {
    "path": "UpdateLog/update_log_en-us.md",
    "content": "**[简体中文](./update_log.md) | [繁體中文](./update_log_zh-tw.md) | English**<br>\n\n# TrafficMonitor Update log\n## V1.83 (2022/02/03)\n\n* Fixed the problem that when there is already a task plan of auto-run, the auto-run at start up will be failed to set.\n* Added the \"Program is already running\" dialog box at the program is started when there is already a TrafficMonitor process running.\n* Added the plug-in information in the mouse tool tips.\n* Fixed an issue where plugins could still be loaded if they were disabled.\n* Added the support of free sorting of plugin items in the taskbar.\n* Added the function to select the network connections to be displayed in the \"Network Connection List\" in the option settings.\n* Added the function of displaying the total network speed.\n* Added the function of displaying the network speed graph in the taskbar.\n* Fixed the problem that only one hard disk can be displayed when there are multiple hard disks with the same name in the hardware monitor.\n* Optimize the display effect of the resource usage graph in plot mode of the taskbar window.\n* Change the \"Display Settings\" in the taskbar window context menu from menu to dialog box.\n* Added \"Refresh connection list\" command to \"Select Network Connections\" menu.\n* Fixed some crashes.\n\n## V1.82 (2021/12/12)\n\n* Added a new plug-in system, you can display more customized content in the taskbar window and the main window through the plug-in.\n* Added the \"Task Manager\" command to the taskbar window context menu.\n* Added the \"Apply\" button to the option settings dialog.\n* Added weekly view in historical traffic statistics.\n* Added the setting of task bar item spacing.\n* Fixed the problem that the taskbar window context menu cannot pop up in the dark mode of Windows 11.\n* Fixed the problem that the taskbar window could not be displayed when the taskbar was restored to the Windows 10 style using software such as StartAllBack under Windows 11.\n\n## V1.81 (2021/07/27)\n\n* Fixed some crashes since version 1.80.\n* Added the support of custom sorting of the items in the taskbar window.\n* When the number of items displayed in the taskbar window is odd, the last item is arranged vertically to save space on the taskbar.\n* Added the function of monitoring the hard disk usage.\n* Add some settings in the right-click menu to the option settings dialog.\n* Added the support for multiple monitors of the main window. You can move the main window to other monitors without checking \"Allow out of screen boundaries\".\n\n## V1.80.3 (2021/05/29)\n\n* Fixed the problem that the CPU temperature cannot be displayed\n* Added the function to select the temperature of specified CPU core to monitor.\n* Fixed a problem that caused the program crashed.\n\n## V1.80.2 (2021/05/22)\n\n* Fixed several issues that cause the program crashed.\n* Use LibreHardwareMonitor to obtain temperature information.\n* Added the function to select the temperature of which hard drive to monitor.\n* Added the function to select the hardware to monitor.\n* Fix the problem that the LibreHardwareMonitorLib.sys file is not released when the program exits.\n\n## V1.80.1 (2021/05/17)\n\n* Fixed the problem that the temperature of AMD GPU could not be obtained.\n* Fixed the problem that the program crashes when automatically switching to the light mode color preset.\n* Fixed the problem that the temperature of the 11th generation Intel processor cannot be obtained.\n* Fixed the problem that the auto-start setting does not work.\n\n## V1.80 (2021/05/15)\n\n* Added the function of monitoring the temperature of CPU, GPU, hard disk and main board.\n* Added the function of monitoring the GPU usage.\n* Fixed the problem that the historical traffic statistics is not correct caused by the data overflowed.\n* Fixed the problem that the main window position was incorrect after the screen resolution was changed.\n* Fixed the problem that the size of the interface elements of the main window and taskbar window would not change after the system DPI was changed.\n* Added the ability to freely specify display items in the taskbar window.\n* Added xml format of the main window skin, supporting temperature display.\n* Fixed the problem that the sub-windows in the option settings cannot be scrolled by the touch screen when using a touch screen device.\n* Put the processing of checking for updates into a background thread to solve the problem that the program does not respond for a long time when checking for update.\n* Fix the problem that the program is abnormal after pressing Alt+F4 in the taskbar window.\n* Added support for net speeds above 4GB per second.\n* Added the option to set the update source as Gitee to solve the problem that the program cannot be updated sometimes due to the inaccessibility of GitHub in mainland China.\n* Added setting of memory display mode.\n* Fix the problem that the currently monitored network connection is closed or disabled, the previously monitored network connection cannot be automatically selected after restarted.\n* Auto run at boot is realized by Task Scheduler to solve the problem that the auto run at boot dose not work sometimes.\n* Fix some crashes.\n* Improvements in other details.\n\n## V1.79.1 (2020/08/05)\n\n* Fix the problem of registry handle leak.\n* Fixed the problem that CPU and memory usage are not counted when the main window and taskbar window are not displayed.\n\n## V1.79 (2020/07/30)\n\n* Add the function of automatically adapts to the Windows 10 dark/light theme of the taskbar window color.\n* Add the function of automatically adapts to the Windows 10 dark/light theme of the notification area icon.\n* Add the option of CPU usage acquisition method. Fix the problem that the CPU usage may always 0 of some users.\n* Remove the option of \"Transparent color\" in \"Options\"-\"Taskbar Window Settings\", and add the option of \"Background Color Transparent\".\n* Allows only CPU and memory usage displayed in the taskbar window without the net speed.\n* Fixes an problem that the settings may not be saved after the shutdown.\n* Add the setting of \"the first day of the week\" in the \"Historical Traffic Statistics\"-\"Calendar View\".\n* Add the function of displaying history traffic by day, month, quarter and year in the \"Historical Traffic Statistics\"-\"List View\".\n* May fixed the problem of the loss of historical traffic data in small probability.\n* Fixed the problem that the right-click menu cannot pop up if the taskbar window background is set to transparent in Windows 10 light theme.\n* Fixed the problem that traffic statistics does not support data exceeding 2TB.\n* Added icons for menu items.\n* Save the data to the AppData directory when the program's directory cannot be written.\n* Add the function of setting the monitoring interval.\n\n## V1.78 (2020/03/21)\n\n* Add the function to open the specified application by double-clicking the main window or taskbar window.\n* Add the function to display the CPU and memory utilization status bar in the taskbar window.\n* Fix the inconsistence of CPU usage with the task manager in Windows 10.\n* Added option to show mouse tool tips.\n* Added the function to set the taskbar color according to Windows 10 light mode when the program is first started.\n* Added the function of preset schemes in the taskbar settings.\n* Improvements in other details.\n\n## V1.77 (2019/05/01)\n* Add the transparent color setting of the task window. Fix the problem that the text color of the taskbar window cannot be set to black when the taskbar is white (Set the transparent color in \"Options\" - \"Taskbar Window Settings\").\n* Add the function to display crash information when the program crashes.\n* Fix the problem that the screen will flicker when the Windows Explorer restarts if the taskbar window is displayed.\n* Display monitoring information when the mouse points to the notification area icon.\n* Fixed the problem that the network speed could not be displayed when using a Bluetooth network.\n* Added x64 version.\n* Improvements in other details.\n### Release Notes:\nThis update solves the problem that the text is not clear in the taskbar when the taskbar is white in the latest version of Windows 10. You need to manually set the transparent color in \"Options\" - \"Taskbar Window Settings\" - \"Transparent Color\". It is an opaque effect when the transparent color is not black and is different from the background color, and is transparent if it is the same as the background color. In the Windows 10 white taskbar, it is recommended to set as the following picture:\n![白色任务栏截图](https://user-images.githubusercontent.com/30562462/57004858-36b55300-6c05-11e9-89d8-9911dc99f09c.PNG)\n## V1.76 (2018/11/11)\n* Fix the problem that the resolution changes may cause the program crash.\n* Add the options for the saving location of the configuration files.\n* Fix the problem that the vertical position of the taskbar window was incorrect when the resolution changed.\n* Add the display of today’s upload and download traffic. Add the upload and download traffic statics in the history traffic static dialog.\n* Improvements in other details.\n"
  },
  {
    "path": "UpdateLog/update_log_zh-tw.md",
    "content": "**[简体中文](./update_log.md) | 繁體中文 | [English](./update_log_en-us.md)**<br>\n\n# TrafficMonitor 更新日誌\n## V1.82 (2021/12/12)\n\n* 修正當已經存在一個開機自啟動的任務計劃時，無法設定開機自啟動的問題，\n* 新增當已經存在TrafficMonitor程序時啟動程式，彈出“程式已經在執行”對話方塊。\n* 滑鼠提示中新增外掛資訊的顯示。\n* 修正外掛被禁用後仍然會被載入的問題。\n* 工作列中的外掛專案支援自由排序。\n* 選項設定中新增可以選擇要在“網路連線列表”中顯示的網路的功能。\n* 新增顯示總網速的功能。\n* 新增在工作列中顯示網速佔用圖的功能。\n* 修正硬體監控中有多個名稱相同的硬碟時只能顯示一個硬碟的問題。\n* 最佳化工作列視窗橫向滾動圖的顯示效果。\n* 工作列視窗右鍵選單中的“顯示設定”由選單改為對話方塊形式。\n* “選擇網路連線”選單中新增“重新整理網路連線列表”命令。\n* 修正幾處導致程式崩潰的問題。\n\n## V1.82 (2021/12/12)\n\n* 新增外掛系統，可以透過外掛在工作列視窗和主視窗顯示更多自定義內容。\n* 工作列右鍵選單中增加“工作管理員”命令。\n* 選項設定中新增“應用”按鈕。\n* 歷史流量統計中增加周檢視。\n* 新增工作列專案間距的設定。\n* 修正Windows11深色模式下工作列右鍵選單無法彈出的問題。\n* 修正Windows11下使用StartAllBack等軟體將工作列恢復為Windows10樣式時，工作列視窗無法顯示的問題。\n\n## V1.81 (2021/07/27)\n\n* 修正1.80版本以來的一些崩潰的問題\n* 工作列視窗顯示專案支援自定義排序\n* 工作列視窗中顯示的專案數量為奇數時，最後一個專案垂直排列，以節省工作列空間\n* 新增顯示硬碟利用率的功能\n* 將右鍵選單中的一些設定新增到選項設定介面中\n* 新增主視窗對多顯示器的支援，在不勾選“允許超出螢幕邊界”的情況下也能移動到其他顯示器了\n\n## V1.80.3 (2021/05/29)\n\n* 修正無法顯示CPU溫度的問題\n* 新增可以選擇監控指定CPU核心的溫度的功能\n* 修正一處導致程式崩潰的問題\n\n## V1.80.2 (2021/05/22)\n\n* 修正幾處導致程式崩潰的問題\n* 獲取溫度資訊改用LibreHardwareMonitor\n* 新增可選擇要監控哪塊硬碟的溫度的功能\n* 新增可選擇要監控哪些硬體的功能\n* 修正程式退出時LibreHardwareMonitorLib.sys檔案沒有解除佔用的問題\n\n## V1.80.1 (2021/05/17)\n\n* 修正無法獲取AMD顯示卡溫度的問題\n* 修正自動切換為淺色模式顏色預設時程式崩潰的問題\n* 修正無法獲取11代Intel處理器溫度的問題\n* 修正設定開機自啟動無效的問題\n\n## V1.80 (2021/05/15)\n\n* 新增CPU、顯示卡、硬碟和主機板溫度監控的功能\n* 新增顯示卡利用率監控的功能\n* 修正歷史流量統計數值溢位導致統計數值不正確的問題\n* 修正螢幕解析度更改後主視窗位置不正確的問題\n* 修正系統DPI更改後，主視窗和工作列視窗介面元素的大小不會隨DPI變化的問題\n* 新增工作列視窗自由指定顯示專案的功能\n* 新增xml格式的主視窗面板，支援溫度顯示\n* 修正使用觸屏裝置時，選項設定中的子視窗無法使用觸屏滾動的問題\n* 將檢查更新的處理放到後臺執行緒中，以解決檢查更新時程式長時間沒有響應的問題\n* 修正工作列視窗中按Alt+F4後程序異常的問題\n* 新增對每秒4GB以上網速的支援\n* 新增可選擇更新源為Gitee，以解決中國大陸地區有時無法訪問GitHub導致程式無法更新的問題\n* 新增記憶體顯示方式設定\n* 修正當前監控的網絡卡被關閉或禁用，再次啟動後無法自動選擇之前監控的網絡卡的問題\n* 開機自啟動採用任務計劃實現，以解決有時開機自啟動無效的問題\n* 修正一些崩潰的問題\n* 其他細節方面的改進\n\n## V1.79.1 (2020/08/05)\n\n* 修正登錄檔控制碼洩漏的問題。\n* 修正當主視窗和工作列視窗都不顯示時，不統計CPU和記憶體利用率的問題。\n\n## V1.79 (2020/07/30)\n\n* 新增工作列視窗顏色自動適應Windows10深色/淺色主題的功能。\n* 新增通知區圖示自動適應Windows10深色/淺色主題的功能。\n* 增加CPU獲取方式的設定，解決部分使用者出現的CPU利用率始終為0的問題。\n* 選項設定>工作列設定中去掉「透明色」選項，新增「背景色透明」的選項。\n* 新增允許工作列只顯示CPU和記憶體利用量而不顯示網速。\n* 修正了關機後設置可能會沒有儲存的問題。\n* 歷史流量統計-日曆檢視中增加了每週第一天的設定。\n* 歷史流量統計-列表檢視中增加了按年、月、日、季度統計的功能。\n* 可能解決了歷史流量資料有小概率丟失的問題。\n* 修正Windows10淺色主題時，如果工作列視窗背景設定為透明，會無法彈出右鍵選單的問題。\n* 修正流量統計不支援統計超過2TB的問題。\n* 為選單項添加了圖示。\n* 當程式所在目錄無法寫入時，將資料儲存到AppData目錄。\n* 新增設定監視時間間隔的功能。\n\n## V1.78 (2020/03/21)\n\n* 新增按兩下主視窗或工作列視窗開啟指定應用程式的功能\n* 新增工作列視窗中顯示CPU和RAM可用狀態條的功能\n* 修正 Windows 10 中 CPU 使用量和工作管理員不一致的問題\n* 新增是否顯示游標提示的選項\n* 新增程式首次啟動時根據 Windows 10 淺色模式設定工作列顏色的功能\n* 工作列設定中增加預設方案的功能\n* 其他細節方面的改進\n\n## V1.77 (2019/05/01)\n* 新增工作列視窗透明色設定，修正工作列為白色時工作列視窗文字顏色無法設定為黑色的問題（在 [選項] —— [工作列視窗設定] 中設定透明色）\n* 新增程式崩潰時顯示崩潰資訊的功能\n* 修正顯示工作列視窗的情况下，檔案總管重新啟動時會導致螢幕畫面閃爍的問題\n* 新增滑鼠指向通知區域圖示時顯示監控資訊\n* 修正使用藍牙網路時無法顯示網速的問題\n* 新增x64的版本\n* 其他細節方面的改進\n### 更新說明：\n本次更新在一定程度上解決了Win10最新版本中白色工作列時文字看不清的問題。需要手動在“選項”——“工作列視窗設定”——“透明色”設定透明色。當透明色不為黑色且和背景色不同時為不透明效果，如果和背景色相同則為透明效果。在Win10白色工作列中建議如下圖所示設定：\n![白色任务栏截图](https://user-images.githubusercontent.com/30562462/57004858-36b55300-6c05-11e9-89d8-9911dc99f09c.PNG)\n\n## V1.76 (2018/11/11)\n* 修正了解析度更改可能會導致程式崩潰的問題；\n* 新增設定檔儲存位置的選項；\n* 修正了當解析度更改時工作列視窗的垂直位置不正確的問題；\n* 新增今日上傳和下載流量的顯示，歷史流量統計中增加上傳和下載流量的統計；\n* 其他細節方面的改進。\n\n"
  },
  {
    "path": "include/OpenHardwareMonitor/OpenHardwareMonitorApi.h",
    "content": "﻿#pragma once\n#include <memory>\n#include \"OpenHardwareMonitorGlobal.h\"\n#include <map>\n#include <string>\n\nnamespace OpenHardwareMonitorApi\n{\n    class IOpenHardwareMonitor\n    {\n    public:\n        virtual void GetHardwareInfo() = 0;     //获取一次硬件信息\n        virtual float CpuTemperature() = 0;     //返回获取到的CPU温度\n        virtual float GpuTemperature() = 0;     //返回获取到的GPU温度\n        virtual float HDDTemperature() = 0;     //返回获取到的硬盘温度\n        virtual float MainboardTemperature() = 0;   //返回获取到的主板温度\n        virtual float GpuUsage() = 0;           //返回获取到的GPU利用率\n        virtual float CpuFreq() = 0;            //返回获取到的CPU频率\n        virtual const std::map<std::wstring, float>& AllHDDTemperature() = 0;   //返回所有硬盘的温度。map的key是硬盘的名称，value是硬盘的温度\n        virtual const std::map<std::wstring, float>& AllCpuTemperature() = 0;   //返回所有CPU（核心）的温度。map的key是CPU的名称，value是硬盘的温度\n        virtual const std::map<std::wstring, float>& AllHDDUsage() = 0;         //返回所有硬盘的使用率\n\n        virtual void SetCpuEnable(bool enable) = 0;\n        virtual void SetGpuEnable(bool enable) = 0;\n        virtual void SetHddEnable(bool enable) = 0;\n        virtual void SetMainboardEnable(bool enable) = 0;\n    };\n\n    OPENHARDWAREMONITOR_API std::shared_ptr<IOpenHardwareMonitor> CreateInstance();\n    OPENHARDWAREMONITOR_API std::wstring GetErrorMessage();\n}\n"
  },
  {
    "path": "include/OpenHardwareMonitor/OpenHardwareMonitorGlobal.h",
    "content": "﻿#pragma once\n\n#ifdef OPENHARDWAREMONITOR_EXPORTS\n#define OPENHARDWAREMONITOR_API __declspec(dllexport)\n#else\n#define OPENHARDWAREMONITOR_API __declspec(dllimport)\n#endif\n"
  },
  {
    "path": "include/PluginInterface.h",
    "content": "﻿/*********************************************************\n* TrafficMonitor 插件接口\n* Copyright (C) by Zhong Yang 2021\n* zhongyang219@hotmail.com\n**********************************************************/\n#pragma once\n\n//插件显示项目的接口\nclass IPluginItem\n{\npublic:\n    /**\n     * @brief   获取显示项目的名称\n     * @return  const wchar_t*\n     */\n    virtual const wchar_t* GetItemName() const = 0;\n\n    /**\n     * @brief   获取显示项目的唯一ID\n     * @return  const wchar_t*\n     */\n    virtual const wchar_t* GetItemId() const = 0;\n\n    /**\n     * @brief   获取项目标签的文本\n     * @return  const wchar_t*\n     */\n    virtual const wchar_t* GetItemLableText() const = 0;\n\n    /**\n     * @brief   获取项目数值的文本\n     * @detail  由于此函数可能会被频繁调用，因此不要在这里获取监控数据，\n     *  而是在ITMPlugin::DataRequired函数中获取数据后保存起来，然后在这里返回获取的数值\n     * @return  const wchar_t*\n     */\n    virtual const wchar_t* GetItemValueText() const = 0;\n\n    /**\n     * @brief   获取项目数值的示例文本\n     * @detail  此函数返回的字符串的长度会用于计算显示区域的宽度\n     * @return  const wchar_t*\n     */\n    virtual const wchar_t* GetItemValueSampleText() const = 0;\n\n    /**\n     * @brief   显示区域是否由插件自行绘制\n     * @detail\n     *  如果返回false，则根据GetItemLableText和GetItemValueText返回的文本由主程序绘制显示区域，重写DrawItem函数将不起作用。\n     *  如果重写此函数并返回true，则必须重写DrawItem函数并在里面添加绘制显示区域的代码，\n     *  此时GetItemLableText、GetItemValueText和GetItemValueSampleText的返回值将被主程序忽略\n     * @return  bool\n     */\n    virtual bool IsCustomDraw() const { return false; }\n\n    /**\n     * @brief   获取显示区域的宽度\n     * @detail\n     *  只有当CustomDraw()函数返回true时重写此函数才有效。\n     *  返回的值为DPI为96（100%）时的宽度，主程序会根据当前系统DPI的设置自动按比例放大，\n     *  因此你不需要为不同的DPI设置返回不同的值。\n     *  需要注意的是，这里的返回值代表了自绘区域所需要的最小宽度，DrawItem函数中的参数w的值可能会大于这个值\n     * @return  int\n     */\n    virtual int GetItemWidth() const { return 0; }\n\n    /**\n     * @brief   自定义绘制显示区域的函数，只有当CustomDraw()函数返回true时重写此函数才有效\n     * @param   void * hDC 绘图的上下文句柄\n     * @param   int x 绘图的矩形区域\n     * @param   int y\n     * @param   int w\n     * @param   int h\n     * @param   bool dark_mode 深色模式为true，浅色模式为false\n     * @return  void\n     */\n    virtual void DrawItem(void* hDC, int x, int y, int w, int h, bool dark_mode) {}\n\n    /**\n     * @brief   获取显示区域的宽度\n     * @detail\n     *  只有当CustomDraw()函数返回true时重写此函数才有效。\n     *  此函数和GetItemWidth不同，插件可以根据参数hDC来计算需要的宽度，\n     *  它返回的是实际的宽度，主程序不会根据当前系统的DPI对返回值进行放大。\n     *  需要注意的是，这里的返回值代表了自绘区域所需要的最小宽度，DrawItem函数中的参数w的值可能会大于这个值\n     * @param   void * hDC 绘图的上下文句柄\n     * @return  int\n     */\n    virtual int GetItemWidthEx(void* hDC) const { return 0; }\n\n    /** 鼠标事件的类型 */\n    enum MouseEventType\n    {\n        MT_LCLICKED,    /**< 点击了鼠标左键 */\n        MT_RCLICKED,    /**< 点击了鼠标右键 */\n        MT_DBCLICKED,   /**< 双击了鼠标左键 */\n    };\n\n    enum MouseEventFlag\n    {\n        MF_TASKBAR_WND = 1 << 0,        /**< 是否为任务栏窗口的鼠标事件 */\n    };\n\n    /**\n     * @brief   当插件显示区域有鼠标事件时由主程序调用\n     * @param   MouseEventType type 鼠标事件的类型\n     * @param   int x 鼠标指针所在的x坐标\n     * @param   int y 鼠标指针所在的y坐标\n     * @param   void* hWnd 产生此鼠标事件的窗口的句柄（主窗口或任务栏窗口）\n     * @param   int flag 为若干MouseEventFlag枚举常量的组合\n     * @return  int 如果返回非0，则主程序认为插件已经对此鼠标事件作出了全部的响应，主程序将不会再对此鼠标事件做额外的响应。\n     *   例如当type为MT_RCLICKED时，如果程序返回0，则会弹出主程序提供的右键菜单；而返回非0时，主程序不会再做任何处理。\n     */\n    virtual int OnMouseEvent(MouseEventType type, int x, int y, void* hWnd, int flag) { return 0; }\n\n    enum ItemInfoType\n    {\n\n    };\n    virtual void* OnItemInfo(ItemInfoType, void* para1, void* para2) { return 0; }\n};\n\n\n//插件接口\nclass ITMPlugin\n{\npublic:\n    /**\n     * @brief   插件接口的版本，仅当修改了插件接口时才会修改这里的返回值。\n     * @attention 插件开发者不应该修改这里的返回值，也不应该重写此虚函数。\n     * @return  int\n     */\n    virtual int GetAPIVersion() const { return 3; }\n\n    /**\n     * @brief   获取插件显示项目的对象\n     * @detail  一个插件dll可以提供多个实现IPluginItem接口的对象，对应多个显示项目。\n     *  当index的值大于或等于0且小于IPluginItem接口的对象的个数时，返回对象的IPluginItem接口的指针，其他情况应该返回空指针。\n     *  例如插件提供两个显示项目，则当index等于0或1时返回对应IPluginItem接口的对象，其他值时必须返回空指针。\n     * @param   int index 对象的索引\n     * @return  IPluginItem* 插件显示项目的对象\n     */\n    virtual IPluginItem* GetItem(int index) = 0;\n\n    /**\n     * @brief   主程序会每隔一定时间调用此函数，插件需要在函数里获取一次监控的数据\n     */\n    virtual void DataRequired() = 0;\n\n    /** 选项设置对话框的返回值 */\n    enum OptionReturn\n    {\n        OR_OPTION_CHANGED,          /**< 选项设置对话框中更改了选项设置 */\n        OR_OPTION_UNCHANGED,        /**< 选项设置对话框中未更改选项设置 */\n        OR_OPTION_NOT_PROVIDED      /**< 未提供选项设置对话框 */\n    };\n\n    /**\n     * @brief   主程序调用此函数以打开插件的选项设置对话框\n     * @detail  此函数不一定要重写。如果插件提供了选项设置界面，则应该重写此函数，并在最后返回OR_OPTION_CHANGED或OR_OPTION_UNCHANGED。\n     * @param   void * hParent 父窗口的句柄\n     *  返回值为OR_OPTION_NOT_PRVIDED则认为插件不提供选项设置对话框。\n     * @return  ITMPlugin::OptionReturn\n     */\n    virtual OptionReturn ShowOptionsDialog(void* hParent) { return OR_OPTION_NOT_PROVIDED; }\n\n    /** 插件信息的索引 */\n    enum PluginInfoIndex\n    {\n        TMI_NAME,           /**< 名称 */\n        TMI_DESCRIPTION,    /**< 描述 */\n        TMI_AUTHOR,         /**< 作者 */\n        TMI_COPYRIGHT,      /**< 版权 */\n        TMI_VERSION,        /**< 版本 */\n        TMI_URL,            /**< 主页 */\n        TMI_MAX             /**< 插件信息的最大值 */\n    };\n\n    /**\n     * @brief   获取此插件的信息，根据index的值返回对应的信息\n     */\n    virtual const wchar_t* GetInfo(PluginInfoIndex index) = 0;\n\n    /** 主程序的监控信息 */\n    struct MonitorInfo\n    {\n        unsigned long long up_speed{};\n        unsigned long long down_speed{};\n        int cpu_usage{};\n        int memory_usage{};\n        int gpu_usage{};\n        int hdd_usage{};\n        int cpu_temperature{};\n        int gpu_temperature{};\n        int hdd_temperature{};\n        int main_board_temperature{};\n        int cpu_freq{};\n    };\n\n    /**\n     * @brief   主程序调用此函数以向插件传递所有获取到的监控信息\n     */\n    virtual void OnMonitorInfo(const MonitorInfo& monitor_info) {}\n\n    /**\n     * @brief   获取插件要在鼠标提示中显示的文本\n     */\n    virtual const wchar_t* GetTooltipInfo() { return L\"\"; }\n\n    enum ExtendedInfoIndex\n    {\n        EI_LABEL_TEXT_COLOR,    //绘图的标签文本颜色\n        EI_VALUE_TEXT_COLOR,    //绘图的数值文本颜色\n        EI_DRAW_TASKBAR_WND,    //是否绘制任务栏窗口\n\n        //主窗口选项设置\n        EI_NAIN_WND_NET_SPEED_SHORT_MODE,   //网速显示简洁模式\n        EI_MAIN_WND_SPERATE_WITH_SPACE,     //数值和单位使用空格分隔\n        EI_MAIN_WND_UNIT_BYTE,              //网速单位是否使用B（字节）\n        EI_MAIN_WND_UNIT_SELECT,            //网速单位选择（0：自动，1：固定为KB/s，2：固定为MB/s）\n        EI_MAIN_WND_NOT_SHOW_UNIT,          //不显示网速单位\n        EI_MAIN_WND_NOT_SHOW_PERCENT,       //不显示百分号\n\n        //任务栏窗口设置\n        EI_TASKBAR_WND_NET_SPEED_SHORT_MODE,    //网速显示简洁模式\n        EI_TASKBAR_WND_SPERATE_WITH_SPACE,      //数值和单位使用空格分隔\n        EI_TASKBAR_WND_VALUE_RIGHT_ALIGN,       //数值右对齐\n        EI_TASKBAR_WND_NET_SPEED_WIDTH,         //网速数据宽度\n        EI_TASKBAR_WND_UNIT_BYTE,               //网速单位是否使用B（字节）\n        EI_TASKBAR_WND_UNIT_SELECT,             //网速单位选择（0：自动，1：固定为KB/s，2：固定为MB/s）\n        EI_TASKBAR_WND_NOT_SHOW_UNIT,           //不显示网速单位\n        EI_TASKBAR_WND_NOT_SHOW_PERCENT,        //不显示百分号\n\n        EI_CONFIG_DIR,                      //配置文件的目录\n    };\n\n    /**\n     * @brief   主程序调用此函数以向插件传递更多信息\n     * @param   ExtendedInfoIndex index 信息的索引，用于区分向插件传递的信息\n     * @param   const wchar_t* data 传递的数据\n     * @return  void\n     */\n    virtual void OnExtenedInfo(ExtendedInfoIndex index, const wchar_t* data) {}\n\n};\n\n/*\n* 注意：插件dll需导出以下函数\n* ITMPlugin* TMPluginGetInstance();\n*   函数返回一个ITMPlugin接口的对象的指针。\n*   此对象通常应该为一个全局或静态的对象，在程序运行结束前，它都不应该被释放。\n*/\n\n\n/*\n* 更新记录：\n* -------------------------------------------------------------------------\n* API version |                       更新内容\n* -------------------------------------------------------------------------\n*     1       | 第一个版本\n* -------------------------------------------------------------------------\n*     2       | 新增 ITMPlugin::GetTooltipInfo 函数\n* -------------------------------------------------------------------------\n*     3       | 新增 IPluginItem::GetItemWidthEx, IPluginItem::OnMouseEvent 函数\n* -------------------------------------------------------------------------\n*/\n"
  },
  {
    "path": "version.info",
    "content": "<version>1.83</version>\n<link>https://gitee.com/zhongyang219/TrafficMonitor/attach_files/958687/download/TrafficMonitor_V1.83_x86_Lite.zip</link>\n<link_x64>https://gitee.com/zhongyang219/TrafficMonitor/attach_files/958688/download/TrafficMonitor_V1.83_x64_Lite.zip</link_x64>\n<contents>Ѿһƻʱ޷ÿ⣬\\nѾTrafficMonitorʱ򣬵ѾСԻ\\nʾϢʾ\\núȻᱻص⡣\\nеĲĿ֧\\nѡѡҪڡбʾĹܡ\\nʾٵĹܡ\\nʾռͼĹܡ\\nӲжͬӲʱֻʾһӲ̵⡣\\nŻںͼʾЧ\\nҼ˵еġʾáɲ˵ΪԻʽ\\nѡӡ˵ӡˢб\\n³⡣</contents>"
  },
  {
    "path": "version_utf8.info",
    "content": "﻿<version>1.83</version>\n<GitHub>\n\t<link>https://github.com/zhongyang219/TrafficMonitor/releases/download/V1.83/TrafficMonitor_V1.83_x86.zip</link>\n\t<link_x64>https://github.com/zhongyang219/TrafficMonitor/releases/download/V1.83/TrafficMonitor_V1.83_x64.zip</link_x64>\n\t<link_without_temperature>https://github.com/zhongyang219/TrafficMonitor/releases/download/V1.83/TrafficMonitor_V1.83_x86_Lite.zip</link_without_temperature>\n\t<link_without_temperature_x64>https://github.com/zhongyang219/TrafficMonitor/releases/download/V1.83/TrafficMonitor_V1.83_x64_Lite.zip</link_without_temperature_x64>\n</GitHub>\n<Gitee>\n\t<link>https://gitee.com/zhongyang219/TrafficMonitor/attach_files/958690/download/TrafficMonitor_V1.83_x86.zip</link>\n\t<link_x64>https://gitee.com/zhongyang219/TrafficMonitor/attach_files/958689/download/TrafficMonitor_V1.83_x64.zip</link_x64>\n\t<link_without_temperature>https://gitee.com/zhongyang219/TrafficMonitor/attach_files/958687/download/TrafficMonitor_V1.83_x86_Lite.zip</link_without_temperature>\n\t<link_without_temperature_x64>https://gitee.com/zhongyang219/TrafficMonitor/attach_files/958688/download/TrafficMonitor_V1.83_x64_Lite.zip</link_without_temperature_x64>\n</Gitee>\n<update_contents>\n\t<contents_zh_cn>修正当已经存在一个开机自启动的任务计划时，无法设置开机自启动的问题，\\n新增当已经存在TrafficMonitor进程时启动程序，弹出“程序已经在运行”对话框。\\n鼠标提示中新增插件信息的显示。\\n修正插件被禁用后仍然会被加载的问题。\\n任务栏中的插件项目支持自由排序。\\n选项设置中新增可以选择要在“网络连接列表”中显示的网络的功能。\\n新增显示总网速的功能。\\n新增在任务栏中显示网速占用图的功能。\\n修正硬件监控中有多个名称相同的硬盘时只能显示一个硬盘的问题。\\n优化任务栏窗口横向滚动图的显示效果。\\n任务栏窗口右键菜单中的“显示设置”由菜单改为对话框形式。\\n“选择网络连接”菜单中添加“刷新网络连接列表”命令。\\n修正几处导致程序崩溃的问题。</contents_zh_cn>\n\t<contents_en>Fixed the problem that when there is already a task plan of auto-run, the auto-run at start up will be failed to set.\\nAdded the \"Program is already running\" dialog box at the program is started when there is already a TrafficMonitor process running.\\nAdded the plug-in information in the mouse tool tips.\\nFixed an issue where plugins could still be loaded if they were disabled.\\nAdded the support of free sorting of plugin items in the taskbar.\\nAdded the function to select the network connections to be displayed in the \"Network Connection List\" in the option settings.\\nAdded the function of displaying the total network speed.\\nAdded the function of displaying the network speed graph in the taskbar.\\nFixed the problem that only one hard disk can be displayed when there are multiple hard disks with the same name in the hardware monitor.\\nOptimize the display effect of the resource usage graph in plot mode of the taskbar window.\\nChange the \"Display Settings\" in the taskbar window context menu from menu to dialog box.\\nAdded \"Refresh connection list\" command to \"Select Network Connections\" menu.\\nFixed some crashes.\\n</contents_en>\n\t<contents_zh_tw>修正當已經存在一個開機自啟動的任務計劃時，無法設定開機自啟動的問題，\\n新增當已經存在TrafficMonitor程序時啟動程式，彈出“程式已經在執行”對話方塊。\\n滑鼠提示中新增外掛資訊的顯示。\\n修正外掛被禁用後仍然會被載入的問題。\\n工作列中的外掛專案支援自由排序。\\n選項設定中新增可以選擇要在“網路連線列表”中顯示的網路的功能。\\n新增顯示總網速的功能。\\n新增在工作列中顯示網速佔用圖的功能。\\n修正硬體監控中有多個名稱相同的硬碟時只能顯示一個硬碟的問題。\\n最佳化工作列視窗橫向滾動圖的顯示效果。\\n工作列視窗右鍵選單中的“顯示設定”由選單改為對話方塊形式。\\n“選擇網路連線”選單中新增“重新整理網路連線列表”命令。\\n修正幾處導致程式崩潰的問題。</contents_zh_tw>\n</update_contents>"
  },
  {
    "path": "皮肤制作教程.md",
    "content": "**皮肤制作教程已经移至Wiki页面：**\n\n[皮肤制作教程 · zhongyang219/TrafficMonitor Wiki (github.com)](https://github.com/zhongyang219/TrafficMonitor/wiki/皮肤制作教程)\n\n"
  }
]