修改Android Studio项目配置JDK路径和项目Gradle路径的GUI工具

概述

本工具提供了一个基于Python Tkinter的图形用户界面(GUI),用于帮助用户搜索并更新Android Studio项目中的config.properties文件里的java.home路径,以及workspace.xml文件中的last_opened_file_path路径。该工具旨在简化手动查找和编辑这些配置文件的过程,提高工作效率。

系统要求

  • Python 3.x
  • Tkinter库(通常随Python一起安装)

安装步骤

  1. 确保你的系统上已经安装了Python 3.x。
  2. 将上述提供的Python脚本保存到一个文件中,例如命名为config_editor.py
  3. 打开命令行工具(如CMD、PowerShell或终端)。
  4. 导航到包含config_editor.py文件的目录。
  5. 运行脚本:
     
    import tkinter as tk
    from tkinter import scrolledtext
    import os
    import xml.etree.ElementTree as ET
    import json# 定义查找配置配置文件的函数
    def find_properties_files(directory):found_files = []for root, dirs, files in os.walk(directory):for file in files:if file.lower() == 'config.properties':found_files.append(os.path.join(root, file))return found_files# 定义读取java.home的函数
    def read_java_home(file_path):try:with open(file_path, 'r', encoding='utf-8') as file:for line in file:if line.lower().startswith('java.home='):return line.strip()except Exception as e:log_text.insert(tk.END, f"读取配置文件 {file_path} 失败: {e}\n")return None# 定义写入java.home的函数
    def write_java_home(file_path, new_line):try:with open(file_path, 'r+', encoding='utf-8') as file:lines = file.readlines()file.seek(0)file.truncate()found = Falsefor line in lines:if line.lower().startswith('java.home='):file.write(new_line + '\n')found = Trueelse:file.write(line)if not found:file.write(new_line + '\n')except Exception as e:log_text.insert(tk.END, f"写入配置文件 {file_path} 失败: {e}\n")# 搜索按钮点击事件
    def on_search():directory = search_entry.get().strip()if not directory:directory = os.getcwd()log_text.delete(1.0, tk.END)found_files = find_properties_files(directory)if not found_files:status_label.config(text="没有找到任何config.properties配置文件")log_text.insert(tk.END, "没有找到任何config.properties配置文件\n")else:status_label.config(text=f"找到 {len(found_files)} 个config.properties配置文件")log_text.insert(tk.END, f"找到 {len(found_files)} 个config.properties配置文件\n")for file_path in found_files:log_text.insert(tk.END, f"处理配置文件: {file_path}\n")# 更新按钮点击事件
    def on_update():new_value = entry_new_value.get().strip()if not new_value:#log_text.insert(tk.END, "新的java.home路径不能为空\n")status_label.config(text=f"新的java.home路径不能为空")returnlog_text.delete(1.0, tk.END)directory = search_entry.get().strip()if not directory:directory = os.getcwd()log_text.insert(tk.END, f"目标路径: {directory}\n")found_files = find_properties_files(directory)if not found_files:status_label.config(text="没有找到任何config.properties配置文件")log_text.insert(tk.END, "没有找到任何config.properties配置文件\n")else:status_label.config(text=f"找到 {len(found_files)} 个config.properties配置文件")log_text.insert(tk.END, f"找到 {len(found_files)} 个config.properties配置文件\n")for file_path in found_files:log_text.insert(tk.END, f"处理配置文件: {file_path}\n")current_java_home = read_java_home(file_path)if current_java_home:log_text.insert(tk.END, f"配置文件路径: {file_path}, 当前值: {current_java_home}, 已更新: java.home={new_value}\n")else:log_text.insert(tk.END, f"配置文件路径: {file_path}, 未找到java.home属性, 已更新: java.home={new_value}\n")write_java_home(file_path, f"java.home={new_value}")# 搜索workspace.xml配置文件
    def on_workspace_xml_search():directory = search_entry.get().strip()if not directory:directory = os.getcwd()log_text.delete(1.0, tk.END)found_files = []for root, dirs, files in os.walk(directory):for file in files:if file.lower() == 'workspace.xml':found_files.append(os.path.join(root, file))if not found_files:log_text.insert(tk.END, "没有找到workspace.xml配置文件\n")else:status_label.config(text=f"找到 {len(found_files)} 个workspace.xml配置文件")log_text.insert(tk.END, f"找到 {len(found_files)} 个workspace.xml配置文件\n")for file_path in found_files:log_text.insert(tk.END, f"处理配置文件: {file_path}\n")# 更新workspace.xml配置文件
    def on_workspace_xml_update():directory = search_entry.get().strip()if not directory:directory = os.getcwd()log_text.delete(1.0, tk.END)import json
    import os
    import tkinter as tk
    from lxml import etree# 更新workspace.xml配置文件
    def on_workspace_xml_update():directory = search_entry.get().strip()if not directory:directory = os.getcwd()# 获取新的 last_opened_file_pathnew_last_opened_file_path = new_last_opened_file_path_entry.get().strip()if not new_last_opened_file_path:messagebox.showerror("错误", "新的last_opened_file_path不能为空")return# 确保路径使用正斜杠new_last_opened_file_path = new_last_opened_file_path.replace("\\", "/")# 查找workspace.xml配置文件found_files = []for root, dirs, files in os.walk(directory):for file in files:if file.lower() == 'workspace.xml':found_files.append(os.path.join(root, file))if not found_files:messagebox.showinfo("信息", "没有找到workspace.xml配置文件")returnfor workspace_xml_path in found_files:log_text.insert(tk.END, f"处理配置文件: {workspace_xml_path}\n")try:# 读取并解析XML配置文件parser = etree.XMLParser(strip_cdata=False)  # 保持CDATA部分不变tree = etree.parse(workspace_xml_path, parser)root = tree.getroot()# 查找PropertiesComponent组件properties_component = root.find(".//component[@name='PropertiesComponent']")if properties_component is None:log_text.insert(tk.END, "找不到 PropertiesComponent 组件\n")continue# 提取CDATA内容cdata_content = properties_component.textif not cdata_content or not cdata_content.strip().startswith("<![CDATA[") or not cdata_content.strip().endswith("]]>"):log_text.insert(tk.END, "找不到 PropertiesComponent 组件或其 JSON 数据\n")continue# 去掉CDATA标记key_to_string_json = cdata_content.strip()[len("<![CDATA["):-len("]]>")].strip()log_text.insert(tk.END, f"原始 keyToString JSON 数据: {key_to_string_json}\n")# 使用json.loads解析JSON数据key_to_string_dict = json.loads(key_to_string_json, strict=False)# 获取旧的 last_opened_file_pathold_last_opened_file_path = key_to_string_dict.get('keyToString', {}).get('last_opened_file_path', '')log_text.insert(tk.END, f"修改前: last_opened_file_path={old_last_opened_file_path}\n")# 更新last_opened_file_pathkey_to_string_dict.setdefault('keyToString', {})['last_opened_file_path'] = new_last_opened_file_pathlog_text.insert(tk.END, f"修改后: last_opened_file_path={new_last_opened_file_path}\n")# 生成新的 JSON 数据# 保持原始 JSON 数据的格式new_key_to_string_json = json.dumps(key_to_string_dict, ensure_ascii=False, indent=None, separators=(',', ': '))# 构造新的CDATA内容new_cdata_content = "<![CDATA[" + new_key_to_string_json + "]]>"# 更新CDATA内容properties_component.text = new_cdata_content# 写回workspace.xml配置文件tree.write(workspace_xml_path, encoding='utf-8', xml_declaration=True, pretty_print=True)log_text.insert(tk.END, f"已更新或添加last_opened_file_path: {new_last_opened_file_path} 到 {workspace_xml_path}\n")except json.JSONDecodeError as e:log_text.insert(tk.END, f"JSON 解析错误: {e}\n")except Exception as e:log_text.insert(tk.END, f"处理配置文件 {workspace_xml_path} 失败: {e}\n")
    import re
    import os
    import tkinter as tk
    from tkinter import messagebox# 更新workspace.xml配置文件
    def on_workspace_xml_update():directory = search_entry.get().strip()if not directory:directory = os.getcwd()# 获取新的 last_opened_file_pathnew_last_opened_file_path = new_last_opened_file_path_entry.get().strip()if not new_last_opened_file_path:#messagebox.showerror("错误", "新的last_opened_file_path不能为空")status_label.config(text=f"新的last_opened_file_path不能为空")return# 确保路径使用正斜杠new_last_opened_file_path = new_last_opened_file_path.replace("\\", "/")# 查找workspace.xml配置文件found_files = []for root, dirs, files in os.walk(directory):for file in files:if file.lower() == 'workspace.xml':found_files.append(os.path.join(root, file))if not found_files:messagebox.showinfo("信息", "没有找到workspace.xml配置文件")returnfor workspace_xml_path in found_files:# 读取配置文件内容with open(workspace_xml_path, 'r', encoding='utf-8') as file:content = file.read()# 查找包含 "last_opened_file_path" 的 JSON 字符串pattern = r'"last_opened_file_path"\s*:\s*"([^"]+)"'match = re.search(pattern, content)if match:# 替换 "last_opened_file_path" 的值old_value = match.group(1)content = re.sub(pattern, f'"last_opened_file_path": "{new_last_opened_file_path}"', content)# 写回配置文件with open(workspace_xml_path, 'w', encoding='utf-8') as file:file.write(content)log_text.insert(tk.END, f"已将 {old_value} 替换为 {new_last_opened_file_path} 在 {workspace_xml_path}\n")else:log_text.insert(tk.END, f"未找到 last_opened_file_path 在 {workspace_xml_path}\n")import tkinter as tk
    from tkinter import scrolledtext
    import tkinter.font as tkFont
    import os
    import re# RGB值
    background_rgb = (43, 42, 51)
    output_bg_rgb = (31, 31, 31)
    output_fg_rgb = (255, 255, 255)  # 白色
    button_bg_rgb = (31, 31, 31)  # 按钮背景颜色
    button_fg_rgb = (0, 128, 0)  # 按钮文本颜色
    entry_bg_rgb = (31, 31, 31)  # 输入框背景颜色
    entry_fg_rgb = (255, 255, 255)  # 输入框文本颜色
    label_bg_rgb = (43, 42, 51)  # 标签背景颜色
    label_fg_rgb = (255, 255, 255)  # 标签文本颜色# 辅助函数
    def create_label(parent, text, **kwargs):return tk.Label(parent,text=text,font=large_font,fg=f"#{label_fg_rgb[0]:02X}{label_fg_rgb[1]:02X}{label_fg_rgb[2]:02X}",bg=f"#{label_bg_rgb[0]:02X}{label_bg_rgb[1]:02X}{label_bg_rgb[2]:02X}",**kwargs)def create_entry(parent, **kwargs):return tk.Entry(parent,font=large_font,fg=f"#{entry_fg_rgb[0]:02X}{entry_fg_rgb[1]:02X}{entry_fg_rgb[2]:02X}",bg=f"#{entry_bg_rgb[0]:02X}{entry_bg_rgb[1]:02X}{entry_bg_rgb[2]:02X}",**kwargs)def create_button(parent, text, command, **kwargs):return tk.Button(parent,text=text,font=large_font,fg=f"#{button_fg_rgb[0]:02X}{button_fg_rgb[1]:02X}{button_fg_rgb[2]:02X}",bg=f"#{button_bg_rgb[0]:02X}{button_bg_rgb[1]:02X}{button_bg_rgb[2]:02X}",command=command,**kwargs)# 日志记录
    def log(message):log_text.insert(tk.END, message + "\n")log_text.see(tk.END)  # 自动滚动到底部
    # 创建主窗口
    root = tk.Tk()
    # 定义一个更大的字体(不加粗)
    large_font = tkFont.Font(family="宋体", size=13)  # 调整字体大小,但不加粗
    root.title("修改android-studio项目配置JDK路径和项目gradle路径")
    #root.title("Config Properties & Workspace XML Editor")# 设置背景颜色
    root.configure(bg=f"#{background_rgb[0]:02X}{background_rgb[1]:02X}{background_rgb[2]:02X}")# 定义一个更大的字体(不加粗)
    large_font = tkFont.Font(family="宋体", size=13)  # 调整字体大小,但不加粗# 创建GUI组件
    create_label(root, "请输入目录路径 (留空则使用当前路径),搜索config.properties中java.home值").pack(pady=5)
    search_entry = create_entry(root, width=50)
    search_entry.pack(pady=5)# 搜索config.properties按钮
    search_button = create_button(root, "搜索android-studio项目配置JDK路径", on_search)
    search_button.pack(pady=5)# 输入新的java.home值
    create_label(root, "请输入新的java.home值=android-studio项目自定义配置JDK路径").pack(pady=9)
    entry_new_value = create_entry(root, width=50)
    entry_new_value.pack(pady=5)# 更新config.properties按钮
    update_button = create_button(root, "更新android-studio配置JDK路径", on_update)
    update_button.pack(pady=5)# 分隔符
    tk.Frame(height=2, bd=1, relief=tk.SUNKEN).pack(fill=tk.X, padx=5, pady=5)# 搜索workspace.xml按钮
    workspace_xml_search_button = create_button(root, "搜索workspace.xml", on_workspace_xml_search)
    workspace_xml_search_button.pack(pady=5)# 输入新的last_opened_file_path
    create_label(root, "请输入新的last_opened_file_path=你的android-studio的gradle路径").pack(pady=5)
    new_last_opened_file_path_entry = create_entry(root, width=50)
    new_last_opened_file_path_entry.pack(pady=5)# 更新workspace.xml按钮
    workspace_xml_update_button = create_button(root, "更新workspace.xml=配置android-studio项目的gradle路径", on_workspace_xml_update)
    workspace_xml_update_button.pack(pady=5)# 状态标签
    status_label = create_label(root, "")
    status_label.pack(pady=5)# 日志文本框
    log_text = scrolledtext.ScrolledText(root,width=80,height=20,wrap=tk.WORD,font=large_font,fg=f"#{output_fg_rgb[0]:02X}{output_fg_rgb[1]:02X}{output_fg_rgb[2]:02X}",bg=f"#{output_bg_rgb[0]:02X}{output_bg_rgb[1]:02X}{output_bg_rgb[2]:02X}"
    )
    log_text.pack(pady=5)# 运行主循环
    root.mainloop()
     

主窗口布局

  • 输入目录路径:用户可以在这里输入Android Studio项目的根目录路径。如果留空,则默认使用当前工作目录。
  • 搜索config.properties:点击此按钮将搜索指定目录下的config.properties文件,并显示当前的java.home值。
  • 输入新的java.home:用户可以在此输入新的JDK路径。
  • 更新config.properties:点击此按钮将更新config.properties文件中的java.home值为用户指定的新路径。
  • 搜索workspace.xml:点击此按钮将搜索指定目录下的workspace.xml文件,并显示当前的last_opened_file_path值。
  • 输入新的last_opened_file_path:用户可以在此输入新的Gradle路径。
  • 更新workspace.xml:点击此按钮将更新workspace.xml文件中的last_opened_file_path值为用户指定的新路径。
  • 日志文本框:显示操作过程中的信息和结果。

使用步骤

  1. 启动程序:运行config_editor.py后,会弹出一个窗口。
  2. 输入目录路径(可选):在“请输入目录路径 (留空则使用当前路径)”输入框中输入Android Studio项目的根目录路径。如果留空,则默认使用当前工作目录。
  3. 搜索config.properties:点击“搜索android-studio项目配置JDK路径”按钮。程序会在指定目录下查找config.properties文件,并显示当前的java.home值。
  4. 更新java.home
    • 在“请输入新的java.home值”输入框中输入新的JDK路径。
    • 点击“更新android-studio配置JDK路径”按钮,程序会更新config.properties文件中的java.home值。
  5. 搜索workspace.xml:点击“搜索workspace.xml”按钮。程序会在指定目录下查找workspace.xml文件,并显示当前的last_opened_file_path值。
  6. 更新last_opened_file_path
    • 在“请输入新的last_opened_file_path”输入框中输入新的Gradle路径。
    • 点击“更新workspace.xml=配置android-studio项目的gradle路径”按钮,程序会更新workspace.xml文件中的last_opened_file_path值。
  7. 查看日志:所有操作的结果都会显示在日志文本框中,包括成功消息和错误信息。

注意事项

  • 确保你有足够的权限来读取和写入指定的文件。
  • 在更新配置文件之前,请备份原始文件,以防意外发生。
  • 请确保输入的路径是正确的,并且路径存在。
  • 如果路径或文件名包含特殊字符,请确保它们被正确转义或引用。

通过以上步骤,你可以轻松地管理和更新Android Studio项目的配置文件,从而确保项目使用的JDK和Gradle路径是正确的。希望这个工具能帮助你提高开发效率!

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

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

ANSYS Maxwell:3PH 感应电机 - 第 1 部分 - 力与热耦合

在此博客中&#xff0c;我们使用 Ansys RMxprt 创建了 3PH 感应电机的 1D 模型&#xff0c;并从设计中自动开发具有所有设置、边界条件和激励的麦克斯韦模型。 ANSYS RMxprt 1D 模型 - 3PH 感应电机设计 请参阅上一篇博客下面的链接&#xff0c;了解如何设置电机设计的 RMxp…

【linux】网络基础 ---- 数据链路层

用于两个设备(同一种数据链路节点)之间进行传递 数据链路层解决的问题是&#xff1a;直接相连的主机之间&#xff0c;进行数据交付 1. 认识以太网 "以太网" 不是一种具体的网络, 而是一种技术标准&#xff1a; 既包含了数据链路层的内容, 也包含了一些物理层的内容…

递归(二)---力扣22括号生成,力扣78求子集

22. 括号生成https://leetcode.cn/problems/generate-parentheses/ 括号生成 数字 n 代表生成括号的对数&#xff0c;请你设计一个函数&#xff0c;用于能够生成所有可能的并且 有效的 括号组合。 示例 1&#xff1a; 输入&#xff1a;n 3 输出&#xff1a;["((()))&quo…

数据分布之指数分布(sample database classicmodels _No.10)

数据分布之指数分布&#xff08;sample database classicmodels _No.10&#xff09; 准备工作&#xff0c;可以去下载 classicmodels 数据库具体如下 点击&#xff1a;classicmodels 也可以去 下面我的博客资源下载 https://download.csdn.net/download/tomxjc/88685970 文章…

C++语言之类与对象1

什么是类 类是一种抽象的数据类型&#xff0c;作为对象的蓝图或模板&#xff0c;它将具有相同属性和行为的对象进行统一抽象封装。其中属性描述对象的特征&#xff0c;如 “汽车” 类中的颜色、品牌等&#xff1b;方法则定义对象可执行的操作&#xff0c;像 “汽车” 类的启动、…

Elasticsearch 和 Kibana 8.16:Kibana 获得上下文和 BBQ 速度并节省开支!

作者&#xff1a;来自 Elastic Platform Product Team Elastic Search AI 平台&#xff08;Elasticsearch、Kibana 和机器学习&#xff09;的 8.16 版本包含大量新功能&#xff0c;可提高性能、优化工作流程和简化数据管理。 使用更好的二进制量化 (Better Binary Quantizatio…

【Golang】——Gin 框架简介与安装

文章目录 1. Gin 框架概述1.1 什么是 Gin 框架&#xff1f;1.2 为什么选择 Gin&#xff1f;1.3 使用场景 2. 安装 Go 与 Gin 框架2.1 安装 Go 语言环境2.2 初始化 Go 项目2.3 安装 Gin 框架 3. 编写第一个 Gin 应用3.1 Gin 最小化示例代码代码解读3.2 运行程序3.3 测试服务 4. …

RGB与YCbCr转换算法

目录 RGB与YCbCr转换算法RGB与YCbCr色域介绍RGB模型YCbCr色域简介YCbCr的应用YUV 和 YCbCr 的区别 色彩转换公式 RGB 转 YCbCr 实现RGB 转 YCbCr 的 Matlab 实现RGB 转 YCbCr 的 FPGA 实现 YCbCr 转 RGB 实现YCbCr 转 RGB 的 Matlab 实现YCbCr 转 RGB 的 FPGA 实现 RGB与YCbCr转…

WebRTC视频 04 - 视频采集类 VideoCaptureDS 中篇

WebRTC视频 01 - 视频采集整体架构 WebRTC视频 02 - 视频采集类 VideoCaptureModule WebRTC视频 03 - 视频采集类 VideoCaptureDS 上篇 WebRTC视频 04 - 视频采集类 VideoCaptureDS 中篇&#xff08;本文&#xff09; WebRTC视频 05 - 视频采集类 VideoCaptureDS 下篇 一、前言…

MAC上的Office三件套报53错误解决方案(随笔记)

目录 现象原因解决方式1. 可视化2. 命令行 参考链接 现象 最近Mac Mini M4非常热门&#xff0c;我也种草买了一台丐中丐版本来体验一下。 在安装Office三件套后&#xff0c;遇到了一个53的错误&#xff1a; Run-time error 53:File not found: Library/Application Support/A…

人工智能与SEO优化中的关键词策略解析

内容概要 在当今数字化快速发展的时代&#xff0c;人工智能&#xff08;AI&#xff09;与搜索引擎优化&#xff08;SEO&#xff09;的结合正变得愈发重要。关键词策略是SEO优化的一项基础工作&#xff0c;它直接影响到网站的可见性和流量。通过运用智能算法&#xff0c;企业能…

【数据库】如何保证数据库迁移过程中数据的一致性?

在数据库迁移过程中&#xff0c;保证数据的一致性是非常重要的&#xff0c;尤其是在涉及到多个表、多个数据库或分布式系统的情况下。以下是一些确保数据一致性的最佳实践和方法&#xff1a; 1. 备份数据 在开始迁移之前&#xff0c;进行全面的数据备份是确保数据一致性的第…

Kubernetes 10 问,测测你对 k8s 的理解程度

Kubernetes 10 问 假设集群有 2 个 node 节点&#xff0c;其中一个有 pod&#xff0c;另一个则没有&#xff0c;那么新的 pod 会被调度到哪个节点上&#xff1f; 应用程序通过容器的形式运行&#xff0c;如果 OOM&#xff08;Out-of-Memory&#xff09;了&#xff0c;是容器重…

Spring:IoC/DI加载properties文件

Spring框架可以通过Spring的配置文件完成两个数据源druid和C3P0的配置&#xff08;Spring&#xff1a;IOC/DI配置管理第三方bean&#xff09;&#xff0c;但是其中包含了一些问题&#xff0c;我们来分析下: 这两个数据源中都使用到了一些固定的常量如数据库连接四要素&#xf…

时钟之CSS+JS版

写在前面 此版本绘制的时钟基于CSSJS模式。 优点操作简单&#xff0c;缺点当然是不够灵活。下一篇会基于HTML5的canvas标签&#xff0c;使用JS绘制。会更灵活&#xff0c;元素更加丰富。 HTML代码 <div class"box"><article class"clock"><…

云计算虚拟化-kvm创建虚拟机

作者介绍&#xff1a;简历上没有一个精通的运维工程师。希望大家多多关注作者&#xff0c;下面的思维导图也是预计更新的内容和当前进度(不定时更新)。 虚拟化&#xff0c;简单来说就是把一台服务器/PC电脑&#xff0c;虚拟成多台独立的虚拟机&#xff0c;每台虚拟机之间相互隔…

<QNAP 453D QTS-5.x> 日志记录:在 Docker 中运行的 Flask 应用安装 自签名 SSL 证书 解决 Chrome 等浏览器证书安全

原因&#xff1a;Chrome 不信任 ssc 证书 使启用了 HTTPS&#xff0c;即使有使用 自签名证书 (self-signed certificate 非由可信的证书颁发机构 【CA&#xff0c;Certificate Authority】签发的&#xff09;。浏览器 Chrome 默认不信任自签名证书&#xff0c;也会报 NET::ERR_…

模板——实现泛型编程的有力武器

模板——实现泛型编程的有力武器 我们为什么需要模板&#xff1f;模板 前言&#xff1a;关于模板&#xff0c;相信大家都有所而闻&#xff0c;以下是我对C模板的个人看法&#xff0c;希望能够帮助到你们呀&#xff01; 我们为什么需要模板&#xff1f; 请到大家看这一段代码&a…

针对git、giteeVSCode连接的使用 || Live Share插件使用

1.下载git 链接 打开终端&#xff0c;桌面鼠标右键 2.配置密钥 登录gitee。 设置密钥 查看官方文档 跟着教程 复制最后的输出进行密钥添加 验证是否添加成功 3.创建&连接远程仓库 创建仓库 git终端进行配置 远程仓库克隆到本地 桌面终端clone,克隆他人|自己的仓库到本地…

OpenGL ES 文字渲染进阶--渲染中文字体

旧文 OpenGL ES 文字渲染方式有几种? 一文中分别介绍了 OpenGL 利用 Canvas 和 FreeType 绘制文字的方法。 无论采用哪种方式进行渲染,本质上原理都是纹理贴图:将带有文字的图像上传到纹理,然后进行贴图。 渲染中文字体 利用 Canvas 绘制中文字体和绘制其他字体在操作方式上…