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 argparse

from multiprocessing import Event

def 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 = i
        for elem in elem:
            indent(elem, level + 1)
        if not elem.tail or not elem.tail.strip():
            elem.tail = i
    else:
        if level and (not elem.tail or not elem.tail.strip()):
            elem.tail = i
        if not elem.tail:
            elem.tail = "\n"

def add_files_to_group(uvprojx_file_path, mode,folder_path, group_name_target):
    # 改变文件扩展名从 .uvprojx 到 .xml
    base, ext = os.path.splitext(uvprojx_file_path)
    if ext != '.uvprojx':
        print("工程文件扩展名不正确")
        return

    xml_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 = None
            for 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 = group
                    break

            if target_group is None:
                print(f"未发现 '{group_name_target}' 分组,请先创建分组后再尝试")
                # 将文件扩展名改回 .uvprojx
                os.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 = None
            heard_inc = None
            target_header = root.find('.//Cads')
            if target_header == None:
                print("未发现头文件分组Cads")
                return
            else:
                heard_inc = target_header.find('VariousControls')
                if heard_inc == None:
                    print("未发现头文件分组VariousControls")
                    return
                else:
                    heard_inc = heard_inc.find('IncludePath')
                    if heard_inc == None:
                        print("未发现头文件分组IncludePath")
                        return
                    else:
                        print("找到头文件分组")



        #下面没有节点
        if mode == 0 or mode == 1:
            creat_dot = 0 #是否需要创建节点标志,如果有重复则跳过
            init_creat = 0
            file_init = files_node.find('File')
            if file_init == None:
                creat_dot = 1
                init_creat = 1
                print("初始节点为空需要创建节点")

        # 遍历指定文件夹,查找所有 .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 = 0
                                    break
                                else:
                                    if i == len(file_check) - 1:
                                        creat_dot = 1
                                        print("节点不存在,创建节点")
                                    else:
                                       creat_dot = 0
                                    continue
                        if creat_dot == 1:
                            # 创建 File 节点并添加到 Files 节点下
                            file_node = ET.SubElement(files_node, 'File')
                            file_name_node = ET.SubElement(file_node, 'FileName')
                            file_name_node.text = file
                            file_type_node = ET.SubElement(file_node, 'FileType')
                            file_type_node.text = '1'  # .c 文件类型都为 1

                            file_path_node = ET.SubElement(file_node, 'FilePath')
                            file_path_node.text = file_path
                            creat_dot = 0
                            init_creat = 0

        if mode == 0 or mode == 2:
            heard_data = heard_data.rstrip(";") #移除最后一个多加的;
            heard_inc.text = heard_data
            #print(heard_inc.text)

        # 格式化 XML
        indent(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:
        # 将文件扩展名改回 .uvprojx
        os.rename(xml_path, uvprojx_file_path)

#寻找工程文件
def find_uvprojx_file():
    uvprojx_files = glob.glob("*.uvprojx")
    if not uvprojx_files:
        print("未找到工程文件,请把此文件放在keil工程目录下")
        return None
    elif len(uvprojx_files) > 1:
        print("在当前目录中找到多个.uvprojx文件:")
        for i, file in enumerate(uvprojx_files, start=1):
            print(f"{i}. {file}")
        print("请确保目录中只有一个.uvprojx文件")
        return None
    else:
        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()

复制代码

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:/a/699057.html

如若内容造成侵权/违法违规/事实不符,请联系我们进行投诉反馈qq邮箱809451989@qq.com,一经查实,立即删除!

相关文章

【鸿蒙开发教程】HarmonyOS 模块关系梳理

HarmonyOS 梳理模块关系 刚开始开发的时候总是理不清鸿蒙中的模块类型和关系,今天就来梳理下鸿蒙中的模块类型 Module类型 Module按照使用场景可以分为两种类型: ●Ability类型的Module: 用于实现应用的功能和特性。每一个Ability类型的M…

西南交通大学【操作系统实验2】

实验目的 本实验要求学生了解什么是信号,掌握软中断的基本原理;掌握中断信号的使用、进程的创建以及系统计时器的使用。通过对本实验的学习,学生能够学会进程的创建方法,更能加深对Linux中的信号机制的认识,并会使用软…

《银行存量客户运营》导读

前言:在中国生活,没有一个人能够离得开银行,但是又有多少人真正了解银行呢? 通过本书你可以学习到:银行不为外人了解的内部运营机制,甚至可以提前把握银行涨息降息政策规律 银行运营的基础逻辑 “运营”二…

泉城济南的隐秘珍宝与山东旅游必去十大景点

泉城济南的隐秘珍宝与山东旅游必去十大景点 济南,这座历史悠久的城市,不仅以其丰富的人文底蕴著称,还拥有诸多引人入胜的自然景观。在这片华夏神州广阔的齐鲁大地上,济南特别以其“三无风景区”——无影山、无影潭、无影泉——而闻…

ON DUPLICATE KEY UPDATE 子句

ON DUPLICATE KEY UPDATE 是 MySQL 中的一个 SQL 语句中的子句,主要用于在执行 INSERT 操作时处理可能出现的重复键值冲突。当尝试插入的记录导致唯一索引或主键约束冲突时(即试图插入的记录的键值已经存在于表中),此子句会触发一…

neo4j 3.5.5版本创建新的数据库

neo4j 3.5.5版本创建新的数据库 1.找到neo4j的conf文件 点进去 2.点击neo4j.conf 选择记事本打开 3.把graph.db换成自己想要创建的数据库名称 4.打开neo4j服务 出现新的数据库

AI Agent 热门的10篇论文

人工智能代理领域广阔,涵盖广泛的主题,包括多代理系统、强化学习、上下文感知系统以及将大型语言模型 (LLMs) 集成到基于代理的系统中。以下是 arXiv 的一些顶级论文,涵盖了人工智能代理的各个方面: A Framework For Intelligent Multi Agent System Based Neural Network …

rman恢复后,少部分数据文件状态为MISSING000**

客户有套一体机,每天晚上21点开始做rman完全备份,大约第2天上午9点多完成备份,rman备份保留策略保留一份完全备份 6月1日晚21点自动发起备份,6月2日上午10点15分完成备份,并生成了一个控制文件备份 c-4063271871-2024…

量产导入 | KGD 是什么?

文章目录 KGD 是什么?认识KGD定义、功能与应用实例【白话文解析】Known Good「Die」何谓良品裸晶粒 (KGD/KGD Die)?解读KGD产业应用为什么大家纷纷采用KGD? 一窥KGD与芯片封测大趋势 KGD 是什么?认识KGD定义…

【Linux系统】线程与线程控制

本篇博客整理了Linux下线程的概念、线程控制的相关接口,旨在让读者初步认识线程,并为下一篇多线程作铺垫。 目录 一、线程是什么 1.线程是进程的执行流 2.线程的执行、调度、切换 3.页表分级与线程资源分配 4.线程的优缺点 二、线程控制 1.创建…

mmdeploy环境部署流程

参考:mmdeploy/docs/zh_cn/01-how-to-build/linux-x86_64.md at main open-mmlab/mmdeploy (github.com) 从零入门《openmmlab》mmdeploy[1]环境安装及简单上手_哔哩哔哩_bilibili 我的环境: docker容器,ubuntu20.04,cuda11.7…

【万方数据库爬虫简单开发(自用)】

万方数据库爬虫简单开发(自用)(一) 使用Python爬虫实现万方数据库论文的搜索并获取信息1.获取url2.输入关键词3.使用BeautifulSoup解析4.获取文章标题信息 使用Python爬虫实现万方数据库论文的搜索并获取信息 后续会逐步探索更新…

从盛世到衰落,历史上八大强国的兴衰与现代地位!

人类文明史悠久,从远古时代至今日,世界舞台上曾经涌现出许多强盛的帝国。它们在自己的黄金时代,曾经无人能敌,不论是在军事、经济还是文化上都独领风骚。然而,无论多么强大的国家也难逃“兴盛必衰”的命运。今天&#…

javaWeb项目-在线考试系统详细功能介绍

项目关键技术 开发工具:IDEA 、Eclipse 编程语言: Java 数据库: MySQL5.7 框架:ssm、Springboot 前端:Vue、ElementUI 关键技术:springboot、SSM、vue、MYSQL、MAVEN 数据库工具:Navicat、SQLyog 1、Java简介 Java语…

ArcGIS 10.8软件安装包免费下载及安装教程

安装包获取: 【软件名称】:ArcGIS 10.8 【安装包链接 】: 链接:https://pan.quark.cn/s/2240330bf935 提取码:Yixn 【备用链接】: 链接:https://pan.baidu.com/s/13V5o_igcK0suW4SFsWkxeQ?pwdj6kx 提取码…

Springboot 整合 Flowable(一):使用 flowable-UI 绘制流程图

目录 一、Flowable简介 二、Flowable 与 Activiti 的区别 三、流程图的绘制(以员工请假流程图为例) 1、下载 flowable 的压缩包: 2、启动包中的 tomcat 3、登录页面 4、绘制结束,导出 bpmn20.xml文件 一、Flowable简介 Fl…

拥抱AI-图片学习中的卷积神经算法详解

一、定义 卷积神经算法(Convolutional Neural Networks, CNN)是深度学习领域中的一种重要算法,特别适用于处理图像相关的任务。以下是卷积神经算法的详细解释: 1. 基本概念 定义:卷积神经网络是一类包含卷积计算且具…

Oracle的优化器

sql优化第一步:搞懂Oracle中的SQL的执行过程 从图中我们可以看出SQL语句在Oracle中经历了以下的几个步骤: 语法检查:检查SQL拼写是否正确,如果不正确,Oracle会报语法错误。 语义检查:检查SQL中的访问对象…

文件IOoooo

1.1 文件路径 文件路径分为两种: 1、绝对路径:以C:、D:等盘符开头的,就是我们所说的绝对路径,根据它可以直接找到文件的具体位置。 2、相对路径:需要先指定一个目录作为基准目录,从基准目录出发&#xf…

python的resample()函数

介绍 在Python中,resample()函数是一个常用的工具,用于对时间序列数据进行重新采样。这个函数可以将时间序列数据从一个频率转换为另一个频率,比如将每天的数据转换为每月的数据。在本教程中,我将向你展示如何使用resample()函数,并解释每个步骤的具体含义。 整体流程 首先…