mirror of
https://github.com/jimeh/build-emacs-for-macos.git
synced 2026-02-19 13:06:38 +00:00
Compare commits
4 Commits
| Author | SHA1 | Date | |
|---|---|---|---|
|
|
4367d6b80e | ||
|
b5ed3887c1
|
|||
|
607076a91b
|
|||
|
1cda2cc3ff
|
2
.github/.release-please-manifest.json
vendored
2
.github/.release-please-manifest.json
vendored
@@ -1,3 +1,3 @@
|
|||||||
{
|
{
|
||||||
".": "0.6.62"
|
".": "0.6.63"
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -1,5 +1,13 @@
|
|||||||
# Changelog
|
# Changelog
|
||||||
|
|
||||||
|
## [0.6.63](https://github.com/jimeh/build-emacs-for-macos/compare/v0.6.62...v0.6.63) (2025-09-14)
|
||||||
|
|
||||||
|
|
||||||
|
### Features
|
||||||
|
|
||||||
|
* **icon:** add support for custom application icons via --icon-uri option ([607076a](https://github.com/jimeh/build-emacs-for-macos/commit/607076a91bf0f227d16c9404f01a64144290685a))
|
||||||
|
* **icon:** add support for Tahoe icons with --tahoe-icon-uri and --tahoe-icon-name options ([b5ed388](https://github.com/jimeh/build-emacs-for-macos/commit/b5ed3887c172540de4a6190072b1e15f5d5efe2c))
|
||||||
|
|
||||||
## [0.6.62](https://github.com/jimeh/build-emacs-for-macos/compare/v0.6.61...v0.6.62) (2025-09-14)
|
## [0.6.62](https://github.com/jimeh/build-emacs-for-macos/compare/v0.6.61...v0.6.62) (2025-09-14)
|
||||||
|
|
||||||
|
|
||||||
|
|||||||
@@ -141,7 +141,7 @@ Options:
|
|||||||
--[no-]relink-eln-files Enable/disable re-linking shared libraries in bundled *.eln files (default: enabled)
|
--[no-]relink-eln-files Enable/disable re-linking shared libraries in bundled *.eln files (default: enabled)
|
||||||
--[no-]rsvg Enable/disable SVG image support via librsvg (default: enabled)
|
--[no-]rsvg Enable/disable SVG image support via librsvg (default: enabled)
|
||||||
--[no-]dbus Enable/disable dbus support (default: enabled)
|
--[no-]dbus Enable/disable dbus support (default: enabled)
|
||||||
--[no-]alpha-background Enable/disable experimental alpha-background patch when building Emacs 30.x - 31.x (default: disabled)
|
--alpha-background Apply experimental alpha-background patch when building Emacs 30.x - 31.x (default: disabled)
|
||||||
--no-frame-refocus Apply no-frame-refocus patch when building Emacs 27.x - 31.x (default: disabled)
|
--no-frame-refocus Apply no-frame-refocus patch when building Emacs 27.x - 31.x (default: disabled)
|
||||||
--no-titlebar Apply no-titlebar patch when building Emacs 27.x - 28.x (default: disabled)
|
--no-titlebar Apply no-titlebar patch when building Emacs 27.x - 28.x (default: disabled)
|
||||||
--[no-]xwidgets Enable/disable XWidgets when building Emacs 27.x (default: disabled)
|
--[no-]xwidgets Enable/disable XWidgets when building Emacs 27.x (default: disabled)
|
||||||
@@ -155,6 +155,9 @@ Options:
|
|||||||
-o, --output DIR Output directory for finished builds (default: <work-dir>/builds)
|
-o, --output DIR Output directory for finished builds (default: <work-dir>/builds)
|
||||||
--build-name NAME Override generated build name
|
--build-name NAME Override generated build name
|
||||||
--dist-include x,y,z List of extra files to copy from Emacs source into build folder/archive (default: COPYING)
|
--dist-include x,y,z List of extra files to copy from Emacs source into build folder/archive (default: COPYING)
|
||||||
|
--icon-uri URI Local path or URL to a .icns file to replace the default app icon
|
||||||
|
--tahoe-icon-uri URI Local path or URL to an Assets.car file for macOS 26 icons. Requires --tahoe-icon-name.
|
||||||
|
--tahoe-icon-name NAME Name of the icon in Assets.car to set as CFBundleIconName
|
||||||
--[no-]self-sign Enable/disable self-signing of Emacs.app (default: enabled)
|
--[no-]self-sign Enable/disable self-signing of Emacs.app (default: enabled)
|
||||||
--[no-]archive Enable/disable creating *.tbz archive (default: enabled)
|
--[no-]archive Enable/disable creating *.tbz archive (default: enabled)
|
||||||
--[no-]archive-keep-build-dir
|
--[no-]archive-keep-build-dir
|
||||||
|
|||||||
@@ -229,6 +229,14 @@ class Build
|
|||||||
|
|
||||||
handle_native_lisp(app)
|
handle_native_lisp(app)
|
||||||
|
|
||||||
|
if options[:icon_uri] || options[:tahoe_icon_uri] || options[:tahoe_icon_name]
|
||||||
|
IconEmbedder.new(
|
||||||
|
app,
|
||||||
|
icon_uri: options[:icon_uri],
|
||||||
|
tahoe_icon_uri: options[:tahoe_icon_uri],
|
||||||
|
tahoe_icon_name: options[:tahoe_icon_name]
|
||||||
|
).embed
|
||||||
|
end
|
||||||
CLIHelperEmbedder.new(app).embed
|
CLIHelperEmbedder.new(app).embed
|
||||||
CSourcesEmbedder.new(app, @source_dir).embed
|
CSourcesEmbedder.new(app, @source_dir).embed
|
||||||
LibEmbedder.new(
|
LibEmbedder.new(
|
||||||
@@ -1413,6 +1421,98 @@ class CSourcesEmbedder < AbstractEmbedder
|
|||||||
end
|
end
|
||||||
end
|
end
|
||||||
|
|
||||||
|
class IconEmbedder < AbstractEmbedder
|
||||||
|
include Helpers
|
||||||
|
|
||||||
|
def initialize(app, icon_uri: nil, tahoe_icon_uri: nil, tahoe_icon_name: nil)
|
||||||
|
super(app)
|
||||||
|
@icon_uri = icon_uri
|
||||||
|
@tahoe_icon_uri = tahoe_icon_uri
|
||||||
|
@tahoe_icon_name = tahoe_icon_name
|
||||||
|
end
|
||||||
|
|
||||||
|
def embed
|
||||||
|
handle_icns if present?(@icon_uri)
|
||||||
|
handle_tahoe if present?(@tahoe_icon_uri)
|
||||||
|
ensure
|
||||||
|
cleanup_tmpdir
|
||||||
|
end
|
||||||
|
|
||||||
|
private
|
||||||
|
|
||||||
|
def present?(val)
|
||||||
|
!val.nil? && !val.strip.empty?
|
||||||
|
end
|
||||||
|
|
||||||
|
def handle_icns
|
||||||
|
source = resolve_source(@icon_uri, '.icns', 'icon.icns')
|
||||||
|
target = File.join(resources_dir, 'Emacs.icns')
|
||||||
|
info 'Replacing application icon (Emacs.icns)...'
|
||||||
|
run_cmd('cp', '-pRL', source, target)
|
||||||
|
source
|
||||||
|
end
|
||||||
|
|
||||||
|
def handle_tahoe
|
||||||
|
fatal '--tahoe-icon-name is required with --tahoe-icon-uri' \
|
||||||
|
unless present?(@tahoe_icon_name)
|
||||||
|
|
||||||
|
source = resolve_source(@tahoe_icon_uri, '.car', 'Assets.car')
|
||||||
|
target = File.join(resources_dir, 'Assets.car')
|
||||||
|
info 'Placing Tahoe Assets.car into Resources...'
|
||||||
|
run_cmd('cp', '-pRL', source, target)
|
||||||
|
|
||||||
|
set_cf_bundle_icon_name(@tahoe_icon_name)
|
||||||
|
source
|
||||||
|
end
|
||||||
|
|
||||||
|
def set_cf_bundle_icon_name(name)
|
||||||
|
info 'Setting CFBundleIconName in Info.plist...'
|
||||||
|
info_plist = File.join(app, 'Contents', 'Info.plist')
|
||||||
|
fatal "Info.plist not found: #{info_plist}" unless File.exist?(info_plist)
|
||||||
|
|
||||||
|
# Use plutil which adds/replaces the key as needed
|
||||||
|
run_cmd(
|
||||||
|
'plutil', '-replace', 'CFBundleIconName', '-string',
|
||||||
|
name, info_plist
|
||||||
|
)
|
||||||
|
end
|
||||||
|
|
||||||
|
def resolve_source(uri, expected_ext, download_name)
|
||||||
|
file_path = if valid_url?(uri)
|
||||||
|
download_file(uri, download_name)
|
||||||
|
else
|
||||||
|
local = File.expand_path(uri)
|
||||||
|
unless File.exist?(local)
|
||||||
|
fatal "File does not exist: #{local}"
|
||||||
|
end
|
||||||
|
local
|
||||||
|
end
|
||||||
|
|
||||||
|
ext = File.extname(file_path).downcase
|
||||||
|
fatal "Unexpected file type: #{ext} (expected #{expected_ext})" \
|
||||||
|
unless ext == expected_ext
|
||||||
|
|
||||||
|
file_path
|
||||||
|
end
|
||||||
|
|
||||||
|
def tmpdir
|
||||||
|
@tmpdir ||= Dir.mktmpdir(%w[emacs-assets .tmp])
|
||||||
|
end
|
||||||
|
|
||||||
|
def download_file(url, name)
|
||||||
|
path = File.join(tmpdir, name)
|
||||||
|
info "Downloading asset from: #{url}"
|
||||||
|
run_cmd('curl', '-L#', url, '-o', path)
|
||||||
|
path
|
||||||
|
end
|
||||||
|
|
||||||
|
def cleanup_tmpdir
|
||||||
|
return unless @tmpdir
|
||||||
|
|
||||||
|
FileUtils.rm_rf(@tmpdir)
|
||||||
|
end
|
||||||
|
end
|
||||||
|
|
||||||
class LibEmbedder < AbstractEmbedder
|
class LibEmbedder < AbstractEmbedder
|
||||||
attr_reader :lib_sources
|
attr_reader :lib_sources
|
||||||
attr_reader :extra_libs
|
attr_reader :extra_libs
|
||||||
@@ -1907,7 +2007,7 @@ class GccInfo
|
|||||||
def clean_macho_binary(path)
|
def clean_macho_binary(path)
|
||||||
debug "Checking for duplicate RPATHs in #{path}"
|
debug "Checking for duplicate RPATHs in #{path}"
|
||||||
macho_cleaner = MachOCleaner.new(path)
|
macho_cleaner = MachOCleaner.new(path)
|
||||||
return unless macho_cleaner.has_duplicate_rpaths?
|
return unless macho_cleaner.duplicate_rpaths?
|
||||||
|
|
||||||
begin
|
begin
|
||||||
info "Removing duplicate RPATHs from #{path}"
|
info "Removing duplicate RPATHs from #{path}"
|
||||||
@@ -1971,7 +2071,7 @@ class MachOCleaner
|
|||||||
end
|
end
|
||||||
|
|
||||||
# Check if file has duplicate RPATH commands
|
# Check if file has duplicate RPATH commands
|
||||||
def has_duplicate_rpaths?
|
def duplicate_rpaths?
|
||||||
count_duplicate_rpaths(macho_object).positive?
|
count_duplicate_rpaths(macho_object).positive?
|
||||||
end
|
end
|
||||||
|
|
||||||
@@ -2101,6 +2201,14 @@ class CLIOptions
|
|||||||
|
|
||||||
def parse!(args)
|
def parse!(args)
|
||||||
parser.parse!(args)
|
parser.parse!(args)
|
||||||
|
|
||||||
|
if options[:tahoe_icon_uri] &&
|
||||||
|
(
|
||||||
|
options[:tahoe_icon_name].nil? ||
|
||||||
|
options[:tahoe_icon_name].strip.empty?
|
||||||
|
)
|
||||||
|
fatal '--tahoe-icon-name is required when --tahoe-icon-uri is specified'
|
||||||
|
end
|
||||||
rescue OptionParser::InvalidOption => e
|
rescue OptionParser::InvalidOption => e
|
||||||
fatal e.message
|
fatal e.message
|
||||||
end
|
end
|
||||||
@@ -2127,6 +2235,9 @@ class CLIOptions
|
|||||||
github_src_repo: nil,
|
github_src_repo: nil,
|
||||||
github_auth: true,
|
github_auth: true,
|
||||||
dist_include: ['COPYING', 'configure_output.txt'],
|
dist_include: ['COPYING', 'configure_output.txt'],
|
||||||
|
icon_uri: nil,
|
||||||
|
tahoe_icon_uri: nil,
|
||||||
|
tahoe_icon_name: nil,
|
||||||
self_sign: true,
|
self_sign: true,
|
||||||
archive: true,
|
archive: true,
|
||||||
archive_keep: false,
|
archive_keep: false,
|
||||||
@@ -2323,6 +2434,22 @@ class CLIOptions
|
|||||||
'folder/archive (default: COPYING)'
|
'folder/archive (default: COPYING)'
|
||||||
) { |v| options[:dist_include] = v }
|
) { |v| options[:dist_include] = v }
|
||||||
|
|
||||||
|
opts.on(
|
||||||
|
'--icon-uri URI',
|
||||||
|
'Local path or URL to a .icns file to replace the default app icon'
|
||||||
|
) { |v| options[:icon_uri] = v }
|
||||||
|
|
||||||
|
opts.on(
|
||||||
|
'--tahoe-icon-uri URI',
|
||||||
|
'Local path or URL to an Assets.car file for macOS 26 icons. ' \
|
||||||
|
'Requires --tahoe-icon-name.'
|
||||||
|
) { |v| options[:tahoe_icon_uri] = v }
|
||||||
|
|
||||||
|
opts.on(
|
||||||
|
'--tahoe-icon-name NAME',
|
||||||
|
'Name of the icon in Assets.car to set as CFBundleIconName'
|
||||||
|
) { |v| options[:tahoe_icon_name] = v }
|
||||||
|
|
||||||
opts.on(
|
opts.on(
|
||||||
'--[no-]self-sign',
|
'--[no-]self-sign',
|
||||||
'Enable/disable self-signing of Emacs.app (default: enabled)'
|
'Enable/disable self-signing of Emacs.app (default: enabled)'
|
||||||
@@ -2372,7 +2499,7 @@ if __FILE__ == $PROGRAM_NAME
|
|||||||
elsif cli_options[:clean_macho_binary]
|
elsif cli_options[:clean_macho_binary]
|
||||||
macho_cleaner = MachOCleaner.new(cli_options[:clean_macho_binary])
|
macho_cleaner = MachOCleaner.new(cli_options[:clean_macho_binary])
|
||||||
|
|
||||||
if macho_cleaner.has_duplicate_rpaths?
|
if macho_cleaner.duplicate_rpaths?
|
||||||
build.info 'Removing duplicate RPATHs from ' \
|
build.info 'Removing duplicate RPATHs from ' \
|
||||||
"#{cli_options[:clean_macho_binary]}..."
|
"#{cli_options[:clean_macho_binary]}..."
|
||||||
macho_cleaner.clean!
|
macho_cleaner.clean!
|
||||||
|
|||||||
Reference in New Issue
Block a user