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

2024-06-12 04:44

本文主要是介绍Keil一键添加.c文件和头文件路径脚本--可遍历添加整个文件夹,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

最近想移植个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文件也可,命令参考:pyinstaller -F -i .\icon.ico .\keil_add_file.py,放个百度云的链接,里面有我打包好的,不过注意杀毒软件估计会报毒,请添加信任

链接:https://pan.baidu.com/s/1zC7kVboAtQwHZ2Zy5RFmIw?pwd=arzd
提取码:arzd

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

脚本内容如下:

import os
import glob
import xml.etree.ElementTree as ET
import argparsefrom multiprocessing import Eventdef indent(elem, level=0):""" Helper function to indent the XML for pretty printing. """i = "\n" + level * "    "if len(elem):if not elem.text or not elem.text.strip():elem.text = i + "    "if not elem.tail or not elem.tail.strip():elem.tail = ifor elem in elem:indent(elem, level + 1)if not elem.tail or not elem.tail.strip():elem.tail = ielse:if level and (not elem.tail or not elem.tail.strip()):elem.tail = iif not elem.tail:elem.tail = "\n"def add_files_to_group(uvprojx_file_path, mode,folder_path, group_name_target):# 改变文件扩展名从 .uvprojx 到 .xmlbase, ext = os.path.splitext(uvprojx_file_path)if ext != '.uvprojx':print("工程文件扩展名不正确")returnxml_path = base + '.xml'os.rename(uvprojx_file_path, xml_path)try:#解析XML文件tree = ET.parse(xml_path)#获取根节点root = tree.getroot()if mode == 0 or mode == 1:# 找到指定GroupName的Group节点target_group = Nonefor group in root.findall('.//Group'):group_name = group.find('GroupName')if group_name is not None and group_name.text == group_name_target:target_group = groupbreakif target_group is None:print(f"未发现 '{group_name_target}' 分组,请先创建分组后再尝试")# 将文件扩展名改回 .uvprojxos.rename(xml_path, uvprojx_file_path)return# 找到目标 Group 节点下的 Files 节点,如果不存在则创建一个files_node = target_group.find('Files')if files_node is None:files_node = ET.SubElement(target_group, 'Files')#寻找头文件分组if mode == 0 or mode == 2:print("寻找头文件分组......")target_header = Noneheard_inc = Nonetarget_header = root.find('.//Cads')if target_header == None:print("未发现头文件分组Cads")returnelse:heard_inc = target_header.find('VariousControls')if heard_inc == None:print("未发现头文件分组VariousControls")returnelse:heard_inc = heard_inc.find('IncludePath')if heard_inc == None:print("未发现头文件分组IncludePath")returnelse:print("找到头文件分组")#下面没有节点if mode == 0 or mode == 1:creat_dot = 0 #是否需要创建节点标志,如果有重复则跳过init_creat = 0file_init = files_node.find('File')if file_init == None:creat_dot = 1init_creat = 1print("初始节点为空需要创建节点")# 遍历指定文件夹,查找所有 .c 文件if mode == 0 or mode == 2:#print(heard_inc.text)heard_data = heard_inc.text + ";"   #末尾需要先加一个分号for subdir, _, files in os.walk(folder_path):#.h路径if mode == 0 or mode == 2:dir_path = os.path.relpath(subdir, start=os.path.dirname(xml_path))if dir_path in heard_inc.text:print("需要添加的头文件路径已存在本次跳过")else:heard_data = heard_data + dir_path + ";"#.c添加到分组if mode == 0 or mode == 1:for file in files:if file.endswith('.c'):# 计算相对路径file_path = os.path.relpath(os.path.join(subdir, file), start=os.path.dirname(xml_path))#print("路径",file_path)file_check = files_node.findall('File')if init_creat == 0:#print("长度",len(file_check))#遍历当前分组下的节点,检测是否已经包含了该路径,如果有直接跳过for i in range(len(file_check)):if file_path in file_check[i].find("FilePath").text:print("节点已存在本次跳过")creat_dot = 0breakelse:if i == len(file_check) - 1:creat_dot = 1print("节点不存在,创建节点")else:creat_dot = 0continueif creat_dot == 1:# 创建 File 节点并添加到 Files 节点下file_node = ET.SubElement(files_node, 'File')file_name_node = ET.SubElement(file_node, 'FileName')file_name_node.text = filefile_type_node = ET.SubElement(file_node, 'FileType')file_type_node.text = '1'  # .c 文件类型都为 1file_path_node = ET.SubElement(file_node, 'FilePath')file_path_node.text = file_pathcreat_dot = 0init_creat = 0if mode == 0 or mode == 2:heard_data = heard_data.rstrip(";") #移除最后一个多加的;heard_inc.text = heard_data#print(heard_inc.text)# 格式化 XMLindent(root)# 保存修改后的 XML 文件tree.write(xml_path, encoding='utf-8', xml_declaration=True)print("已完成")except ET.ParseError as e:print(f"ParseError: {e}")with open(xml_path, 'r', encoding='utf-8') as file:lines = file.readlines()start = max(0, e.position[0] - 5)end = min(len(lines), e.position[0] + 5)print("Context around the error:")for i in range(start, end):print(f"{i+1}: {lines[i].strip()}")finally:# 将文件扩展名改回 .uvprojxos.rename(xml_path, uvprojx_file_path)#寻找工程文件
def find_uvprojx_file():uvprojx_files = glob.glob("*.uvprojx")if not uvprojx_files:print("未找到工程文件,请把此文件放在keil工程目录下")return Noneelif len(uvprojx_files) > 1:print("在当前目录中找到多个.uvprojx文件:")for i, file in enumerate(uvprojx_files, start=1):print(f"{i}. {file}")print("请确保目录中只有一个.uvprojx文件")return Noneelse:return uvprojx_files[0]if __name__ == "__main__":print("keil一键添加文件和头文件路径脚本\n\需放在keil工程同级目录下\n\参数格式,参数用空格隔开\n\默认模式:0\n\默认路径:\"../../../external/lvgl\"\n\默认分组:\"lvgl\"\n\1.添加模式 0:全部添加(.c文件全添加到分组.h文件夹加入include路径里) 1:只添加.c文件到分组 2:只添加.h文件夹到include里\n\2.要添加的文件夹路径,请使用相对路径\n\3.要添加的分组名称,如果没有分组需要先去keil手动添加分组\n")param = input("请输入参数:")if param:#print(param)args = param.split()args[0] = int(args[0])print(args)else:args = [0,"../../../external/lvgl","lvgl"]print("使用默认参数:",args)uvprojx_file_path = find_uvprojx_file()if uvprojx_file_path:add_files_to_group(uvprojx_file_path, args[0],args[1],args[2])event = Event()event.wait()

复制代码

这篇关于Keil一键添加.c文件和头文件路径脚本--可遍历添加整个文件夹的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/1053232

相关文章

一文教你Python引入其他文件夹下的.py文件

《一文教你Python引入其他文件夹下的.py文件》这篇文章主要为大家详细介绍了如何在Python中引入其他文件夹里的.py文件,并探讨几种常见的实现方式,有需要的小伙伴可以根据需求进行选择... 目录1. 使用sys.path动态添加路径2. 使用相对导入(适用于包结构)3. 使用pythonPATH环境

Linux修改pip和conda缓存路径的几种方法

《Linux修改pip和conda缓存路径的几种方法》在Python生态中,pip和conda是两种常见的软件包管理工具,它们在安装、更新和卸载软件包时都会使用缓存来提高效率,适当地修改它们的缓存路径... 目录一、pip 和 conda 的缓存机制1. pip 的缓存机制默认缓存路径2. conda 的缓

Qt把文件夹从A移动到B的实现示例

《Qt把文件夹从A移动到B的实现示例》本文主要介绍了Qt把文件夹从A移动到B的实现示例,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们下面随着小编来一起学习学... 目录如何移动一个文件? 如何移动文件夹(包含里面的全部内容):如何删除文件夹:QT 文件复制,移动(

Windows系统下如何查找JDK的安装路径

《Windows系统下如何查找JDK的安装路径》:本文主要介绍Windows系统下如何查找JDK的安装路径,文中介绍了三种方法,分别是通过命令行检查、使用verbose选项查找jre目录、以及查看... 目录一、确认是否安装了JDK二、查找路径三、另外一种方式如果很久之前安装了JDK,或者在别人的电脑上,想

Python中Windows和macOS文件路径格式不一致的解决方法

《Python中Windows和macOS文件路径格式不一致的解决方法》在Python中,Windows和macOS的文件路径字符串格式不一致主要体现在路径分隔符上,这种差异可能导致跨平台代码在处理文... 目录方法 1:使用 os.path 模块方法 2:使用 pathlib 模块(推荐)方法 3:统一使

一文教你解决Python不支持中文路径的问题

《一文教你解决Python不支持中文路径的问题》Python是一种广泛使用的高级编程语言,然而在处理包含中文字符的文件路径时,Python有时会表现出一些不友好的行为,下面小编就来为大家介绍一下具体的... 目录问题背景解决方案1. 设置正确的文件编码2. 使用pathlib模块3. 转换路径为Unicod

通过Python脚本批量复制并规范命名视频文件

《通过Python脚本批量复制并规范命名视频文件》本文介绍了如何通过Python脚本批量复制并规范命名视频文件,实现自动补齐数字编号、保留原始文件、智能识别有效文件等功能,听过代码示例介绍的非常详细,... 目录一、问题场景:杂乱的视频文件名二、完整解决方案三、关键技术解析1. 智能路径处理2. 精准文件名

web网络安全之跨站脚本攻击(XSS)详解

《web网络安全之跨站脚本攻击(XSS)详解》:本文主要介绍web网络安全之跨站脚本攻击(XSS)的相关资料,跨站脚本攻击XSS是一种常见的Web安全漏洞,攻击者通过注入恶意脚本诱使用户执行,可能... 目录前言XSS 的类型1. 存储型 XSS(Stored XSS)示例:危害:2. 反射型 XSS(Re

SpringBoot项目注入 traceId 追踪整个请求的日志链路(过程详解)

《SpringBoot项目注入traceId追踪整个请求的日志链路(过程详解)》本文介绍了如何在单体SpringBoot项目中通过手动实现过滤器或拦截器来注入traceId,以追踪整个请求的日志链... SpringBoot项目注入 traceId 来追踪整个请求的日志链路,有了 traceId, 我们在排

Python3脚本实现Excel与TXT的智能转换

《Python3脚本实现Excel与TXT的智能转换》在数据处理的日常工作中,我们经常需要将Excel中的结构化数据转换为其他格式,本文将使用Python3实现Excel与TXT的智能转换,需要的可以... 目录场景应用:为什么需要这种转换技术解析:代码实现详解核心代码展示改进点说明实战演练:从Excel到