首页 > 其他分享 >Keil一键添加.c文件和头文件路径脚本--可遍历添加整个文件夹

Keil一键添加.c文件和头文件路径脚本--可遍历添加整个文件夹

时间:2024-06-08 22:11:46浏览次数:17  
标签:头文件 Keil uvprojx elem heard 添加 file print path

最近想移植个LVGL玩玩,发现文件实在是太多了,加的手疼都没搞完,实在不想搞了就去找脚本和工具,基本没找到一个。。。。。。

主要是自己也懒得去研究写脚本,偶然搜到了一个博主写的脚本,原博客地址:https://blog.csdn.net/riyue2044/article/details/139424599

但是有以下问题:

1.这个脚本的.h文件也加在了分组下面,这样一般是不对的,应该加在Target的C/C++的Include path里面

2.脚本没有重复添加检测,导致如果多次添加,会损坏工程文件

3.输入是命令行式的,使用者可能会忘了参数具体设置

 

之前没接触过XML,python也不熟,所以研究了一下,做以下修改

1.把之前的命令行式的输入改为先运行再输入,会提示具体的参数设置,有默认参数,是以我的工程包来写的

2.把.h文件路径直接加在了Target的C/C++的Include path里面

3.加入文件路径检测,重复添加不会导致文件损坏

4.加入更多提示

5.加入三种模式    0:.c文件和.h路径会一起添加 1:只加.c文件 2:只加.h路径

 

使用方法:需要安装python,或者用python打包成exe文件也可

脚本需放在keil工程目录,需要添加的目录则以相对路径填充,比如"../../../external/lvgl",需要注意的是分组需要提前在keil里面创建好,这个懒得改了,有需要的朋友可以自行修改

脚本内容如下:

  1 import os
  2 import glob
  3 import xml.etree.ElementTree as ET
  4 import argparse
  5 
  6 def indent(elem, level=0):
  7     """ Helper function to indent the XML for pretty printing. """
  8     i = "\n" + level * "    "
  9     if len(elem):
 10         if not elem.text or not elem.text.strip():
 11             elem.text = i + "    "
 12         if not elem.tail or not elem.tail.strip():
 13             elem.tail = i
 14         for elem in elem:
 15             indent(elem, level + 1)
 16         if not elem.tail or not elem.tail.strip():
 17             elem.tail = i
 18     else:
 19         if level and (not elem.tail or not elem.tail.strip()):
 20             elem.tail = i
 21         if not elem.tail:
 22             elem.tail = "\n"
 23 
 24 def add_files_to_group(uvprojx_file_path, mode,folder_path, group_name_target):
 25     # 改变文件扩展名从 .uvprojx 到 .xml
 26     base, ext = os.path.splitext(uvprojx_file_path)
 27     if ext != '.uvprojx':
 28         print("工程文件扩展名不正确")
 29         return
 30     
 31     xml_path = base + '.xml'
 32     os.rename(uvprojx_file_path, xml_path)
 33 
 34     try:
 35         #解析XML文件
 36         tree = ET.parse(xml_path)
 37         #获取根节点
 38         root = tree.getroot()
 39 
 40         if mode == 0 or mode == 1:
 41             # 找到指定GroupName的Group节点
 42             target_group = None
 43             for group in root.findall('.//Group'):
 44                 group_name = group.find('GroupName')
 45                 if group_name is not None and group_name.text == group_name_target:
 46                     target_group = group
 47                     break
 48 
 49             if target_group is None:
 50                 print(f"未发现 '{group_name_target}' 分组,请先创建分组后再尝试")
 51                 # 将文件扩展名改回 .uvprojx
 52                 os.rename(xml_path, uvprojx_file_path)
 53                 return
 54 
 55             # 找到目标 Group 节点下的 Files 节点,如果不存在则创建一个
 56             files_node = target_group.find('Files')
 57             if files_node is None:
 58                 files_node = ET.SubElement(target_group, 'Files')
 59                                 
 60         #寻找头文件分组
 61         if mode == 0 or mode == 2:
 62             print("寻找头文件分组......")
 63             target_header = None
 64             heard_inc = None
 65             target_header = root.find('.//Cads')
 66             if target_header == None:
 67                 print("未发现头文件分组Cads")
 68                 return
 69             else:
 70                 heard_inc = target_header.find('VariousControls')
 71                 if heard_inc == None:
 72                     print("未发现头文件分组VariousControls")
 73                     return
 74                 else:
 75                     heard_inc = heard_inc.find('IncludePath')
 76                     if heard_inc == None:
 77                         print("未发现头文件分组IncludePath")
 78                         return
 79                     else:
 80                         print("找到头文件分组")
 81                        
 82         
 83         
 84         #下面没有节点
 85         if mode == 0 or mode == 1:
 86             creat_dot = 0 #是否需要创建节点标志,如果有重复则跳过 
 87             init_creat = 0
 88             file_init = files_node.find('File')
 89             if file_init == None:
 90                 creat_dot = 1
 91                 init_creat = 1
 92                 print("初始节点为空需要创建节点")
 93   
 94         # 遍历指定文件夹,查找所有 .c 文件  
 95         if mode == 0 or mode == 2:
 96             #print(heard_inc.text)  
 97             heard_data = heard_inc.text   
 98             
 99         for subdir, _, files in os.walk(folder_path):
100             #.h路径
101             if mode == 0 or mode == 2:
102                 dir_path = os.path.relpath(subdir, start=os.path.dirname(xml_path))
103                 if dir_path in heard_inc.text:
104                     print("需要添加的头文件路径已存在本次跳过")
105                 else:
106                     heard_data = heard_data + dir_path + ";"
107                     
108             #.c添加到分组
109             if mode == 0 or mode == 1:
110                 for file in files:     
111                     if file.endswith('.c'):
112                         # 计算相对路径
113                         file_path = os.path.relpath(os.path.join(subdir, file), start=os.path.dirname(xml_path))
114                         #print("路径",file_path)
115                         file_check = files_node.findall('File')
116                         if init_creat == 0:
117                             #print("长度",len(file_check))
118                             #遍历当前分组下的节点,检测是否已经包含了该路径,如果有直接跳过
119                             for i in range(len(file_check)):
120                                 if file_path in file_check[i].find("FilePath").text:
121                                     print("节点已存在本次跳过")
122                                     creat_dot = 0
123                                     break
124                                 else: 
125                                     if i == len(file_check) - 1:
126                                         creat_dot = 1
127                                         print("节点不存在,创建节点")
128                                     else:
129                                        creat_dot = 0
130                                     continue    
131                         if creat_dot == 1:
132                             # 创建 File 节点并添加到 Files 节点下
133                             file_node = ET.SubElement(files_node, 'File')
134                             file_name_node = ET.SubElement(file_node, 'FileName')
135                             file_name_node.text = file
136                             file_type_node = ET.SubElement(file_node, 'FileType')
137                             file_type_node.text = '1'  # .c 文件类型都为 1
138 
139                             file_path_node = ET.SubElement(file_node, 'FilePath')
140                             file_path_node.text = file_path
141                             creat_dot = 0
142                             init_creat = 0
143                             
144         if mode == 0 or mode == 2:
145             heard_data = heard_data.rstrip(";") #移除最后一个多加的;
146             heard_inc.text = heard_data
147             #print(heard_inc.text)                    
148                   
149         # 格式化 XML
150         indent(root)
151 
152         # 保存修改后的 XML 文件
153         tree.write(xml_path, encoding='utf-8', xml_declaration=True)
154         print("已完成")
155 
156     except ET.ParseError as e:
157         print(f"ParseError: {e}")
158         with open(xml_path, 'r', encoding='utf-8') as file:
159             lines = file.readlines()
160             start = max(0, e.position[0] - 5)
161             end = min(len(lines), e.position[0] + 5)
162             print("Context around the error:")
163             for i in range(start, end):
164                 print(f"{i+1}: {lines[i].strip()}")
165 
166     finally:
167         # 将文件扩展名改回 .uvprojx
168         os.rename(xml_path, uvprojx_file_path)
169 
170 #寻找工程文件
171 def find_uvprojx_file():
172     uvprojx_files = glob.glob("*.uvprojx")
173     if not uvprojx_files:
174         print("未找到工程文件,请把此文件放在keil工程目录下")
175         return None
176     elif len(uvprojx_files) > 1:
177         print("在当前目录中找到多个.uvprojx文件:")
178         for i, file in enumerate(uvprojx_files, start=1):
179             print(f"{i}. {file}")
180         print("请确保目录中只有一个.uvprojx文件")
181         return None
182     else:
183         return uvprojx_files[0]
184 
185 if __name__ == "__main__":
186     print("keil一键添加文件和头文件路径脚本\n\
187     需放在keil同级目录下\n\
188     参数格式,参数用空格隔开:不可重复添加\n\
189     默认模式:0\n\
190     路径:\"../../../external/lvgl\"\n\
191     分组:\"lvgl\"\n\
192     1.添加模式 0:全部添加(.c文件全添加到分组.h文件夹加入include路径里) 1:只添加.c文件到分组 2:只添加.h文件夹到include里\n\
193     2.要添加的文件夹路径,请使用相对路径\n\
194     3.要添加的分组名称,如果没有分组需要先去keil手动添加分组\n")
195     
196     param = input("请输入参数:")
197     
198     if param:
199         #print(param)
200         args = param.split()
201         args[0] = int(args[0])
202         print(args)
203     else:
204         args = [0,"../../../external/lvgl","lvgl"]
205         print("使用默认参数:",args)
206     uvprojx_file_path = find_uvprojx_file()
207     if uvprojx_file_path:
208         add_files_to_group(uvprojx_file_path, args[0],args[1],args[2])

 

标签:头文件,Keil,uvprojx,elem,heard,添加,file,print,path
From: https://www.cnblogs.com/xiaoliu1012/p/18238997

相关文章

  • Keil uVersion 4单片机开发指南
    1软件安装双击打开C51V901.exe弹出安装界面,点击Next>>点击同意协议勾选框,接着点击Next>>点击Browse...选择合适的目录,接着点击Next>>按要求填写相关信息,然后点击Next>>软件安装中,等待安装完成点击Finish完成安装2注册激活桌面右键打开KeiluVision4,弹出菜单后选......
  • c++入门笔记——头文件
    【头文件】c++中,一个程序开头必有头文件。头文件有许多个,它们的关系是并列的。<algorithm>:包含STL通用算法。<bitset>:包含bitset类模板。<cassert>:包含断言宏,如assert。<cctype>:包含字符处理函数。<cerrno>:定义错误码变量errno。<cfenv>:提供有关浮点环境的操作。......
  • keil5环境搭建
    一、keil软件安装MDK529--安装包ZIP压缩包--软件注册机1、安装KEIL步骤(****安装路径不要有中文****)点击下一步(Next)选择同意,点击下一步:选择安装路径,参考一下截图,必须全英文注册信息:点击next开始安装:等待安装结束,退出;2、注册软件以管理员的身份运行KEIL软件:......
  • 如何在 Proteus 中添加 esp32 库进行仿真
    前言:    最近有B站粉丝问到如何使用proteus进行ESP32仿真,然后自己去查阅了很多资料进修了一下,整理了出这篇文章,分享一下如何将ESP32库添加到proteus中并进行仿真。  在Proteus中添加ESP32库进行仿真下面具体介绍步骤:1.打开Proteus,转到"库"->"库管理"......
  • SemanticKernel:添加插件
    SemanticKernel介绍SemanticKernel是一个SDK,它将OpenAI、AzureOpenAI和HuggingFace等大型语言模型(LLMs)与C#、Python和Java等传统编程语言集成在一起。SemanticKernel通过允许您定义插件来实现这一点,这些插件可以通过几行代码链接在一起。为什么需要添加插件?大语言模型虽然......
  • VScode右键打开(添加到右键)
    原文链接:https://blog.csdn.net/2301_79279082/article/details/132734873打开注册表:win+R->regedit共需要在三处添加:第一处:1、HKEY_CLASSES_ROOT->*->shell->OpenWithVS(自定义,用户新建)  1.0 右键shell-》新建项-》重命名为OpenWithVS......
  • Zibll子比主题美化WordPress底部添加渐变色统计代码 新增显示评论总数及本周发布文章
    在网站底部添加一个统计信息,可以看到网站的运营情况,如会员数量,文章数量、网站的总浏览量、当天发布的文章数量、网站运营天数等,并且这个统计信息背景是渐变色的,非常漂亮。之前有分享过这个代码,今天新增了一个显示评论总数,把今天发布文章数量我改为了本周发布的数量,如果需要之前......
  • android12 Settings 添加导航栏和状态栏开关
    平台RK3568,android12添加导航栏和状态栏的开关。 通过设置系统属性来默认系统关闭导航栏和状态栏。Index:device/rockchip/rk356x/device.mk===================================================================---device/rockchip/rk356x/device.mk(revision2442......
  • js的 addEventListener如果添加的是相同名称的事件,会被覆盖吗
    在JavaScript中,使用addEventListener方法向元素添加事件监听器时,如果有多个相同的事件名称(比如多次调用addEventListener("click",function)),这些监听器不会互相覆盖,而是会累加。这意味着所有为同一事件类型注册的监听器都会按照添加的顺序依次触发,而不是只有最后一个生效。这......
  • DevExpress WPF中文教程:Grid - 如何向项目添加GridControl并绑定到数据
    DevExpressWPF拥有120+个控件和库,将帮助您交付满足甚至超出企业需求的高性能业务应用程序。通过DevExpressWPF能创建有着强大互动功能的XAML基础应用程序,这些应用程序专注于当代客户的需求和构建未来新一代支持触摸的解决方案。无论是Office办公软件的衍伸产品,还是以数据为中心......