mirror of
https://github.com/zhigang1992/RubyMotion.git
synced 2026-04-29 13:15:37 +08:00
Added support for extensions through the ExtensionTarget
This commit is contained in:
committed by
Eloy Durán
parent
480dcd2e6c
commit
1d681f1680
@@ -23,4 +23,5 @@
|
|||||||
# (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS
|
# (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS
|
||||||
# SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
|
# SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
|
||||||
|
|
||||||
require 'motion/project/target/framework_target'
|
require 'motion/project/target/framework_target'
|
||||||
|
require 'motion/project/target/extension_target'
|
||||||
159
lib/motion/project/target/extension_target.rb
Normal file
159
lib/motion/project/target/extension_target.rb
Normal file
@@ -0,0 +1,159 @@
|
|||||||
|
# encoding: utf-8
|
||||||
|
|
||||||
|
# Copyright (c) 2012, HipByte SPRL and contributors
|
||||||
|
# All rights reserved.
|
||||||
|
#
|
||||||
|
# Redistribution and use in source and binary forms, with or without
|
||||||
|
# modification, are permitted provided that the following conditions are met:
|
||||||
|
#
|
||||||
|
# 1. Redistributions of source code must retain the above copyright notice, this
|
||||||
|
# list of conditions and the following disclaimer.
|
||||||
|
# 2. Redistributions in binary form must reproduce the above copyright notice,
|
||||||
|
# this list of conditions and the following disclaimer in the documentation
|
||||||
|
# and/or other materials provided with the distribution.
|
||||||
|
#
|
||||||
|
# THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS" AND
|
||||||
|
# ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED
|
||||||
|
# WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE
|
||||||
|
# DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT OWNER OR CONTRIBUTORS BE LIABLE FOR
|
||||||
|
# ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES
|
||||||
|
# (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES;
|
||||||
|
# LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND
|
||||||
|
# ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
|
||||||
|
# (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS
|
||||||
|
# SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
|
||||||
|
|
||||||
|
require 'motion/project/builder'
|
||||||
|
|
||||||
|
module Motion; module Project
|
||||||
|
class ExtensionTarget
|
||||||
|
include Rake::DSL if Object.const_defined?(:Rake) && Rake.const_defined?(:DSL)
|
||||||
|
|
||||||
|
attr_accessor :type
|
||||||
|
|
||||||
|
def initialize(path, type, config, opts)
|
||||||
|
@path = path
|
||||||
|
@full_path = File.expand_path(path)
|
||||||
|
@type = type
|
||||||
|
@config = config
|
||||||
|
@opts = opts
|
||||||
|
end
|
||||||
|
|
||||||
|
def build(platform)
|
||||||
|
@platform = platform
|
||||||
|
|
||||||
|
command = if platform == 'iPhoneSimulator'
|
||||||
|
"build:simulator"
|
||||||
|
else
|
||||||
|
if @config.distribution_mode
|
||||||
|
"archive:distribution"
|
||||||
|
else
|
||||||
|
"build:device"
|
||||||
|
end
|
||||||
|
end
|
||||||
|
|
||||||
|
success = system("cd #{@full_path} && bundle exec rake #{command}")
|
||||||
|
unless success
|
||||||
|
App.fail "Target '#{@path}' failed to build"
|
||||||
|
end
|
||||||
|
end
|
||||||
|
|
||||||
|
def copy_products(platform)
|
||||||
|
src_path = src_extension_path
|
||||||
|
dest_path = dest_extension_path
|
||||||
|
FileUtils.mkdir_p(File.join(@config.app_bundle(platform), 'PlugIns'))
|
||||||
|
|
||||||
|
extension_path = File.join(dest_path, extension_name)
|
||||||
|
|
||||||
|
if !File.exist?(extension_path) or File.mtime(src_path) > File.mtime(extension_path)
|
||||||
|
App.info 'Copy', src_path
|
||||||
|
FileUtils.cp_r(src_path, dest_path)
|
||||||
|
|
||||||
|
# Modify CFBundleIdentifier
|
||||||
|
extension_dir = File.join(dest_extension_path, extension_name)
|
||||||
|
info_plist = File.join(extension_dir, 'Info.plist')
|
||||||
|
extension_bundle_name = `/usr/libexec/PlistBuddy -c "print CFBundleName" #{info_plist}`.strip
|
||||||
|
extension_bundle_indentifer = "#{@config.identifier}.#{extension_bundle_name}"
|
||||||
|
`/usr/libexec/PlistBuddy -c "set CFBundleIdentifier #{extension_bundle_indentifer}" #{info_plist}`
|
||||||
|
end
|
||||||
|
end
|
||||||
|
|
||||||
|
def codesign(platform)
|
||||||
|
extension_dir = File.join(dest_extension_path, extension_name)
|
||||||
|
|
||||||
|
# Create bundle/ResourceRules.plist.
|
||||||
|
resource_rules_plist = File.join(extension_dir, 'ResourceRules.plist')
|
||||||
|
unless File.exist?(resource_rules_plist)
|
||||||
|
App.info 'Create', resource_rules_plist
|
||||||
|
File.open(resource_rules_plist, 'w') do |io|
|
||||||
|
io.write(<<-PLIST)
|
||||||
|
<?xml version="1.0" encoding="UTF-8"?>
|
||||||
|
<!DOCTYPE plist PUBLIC "-//Apple//DTD PLIST 1.0//EN" "http://www.apple.com/DTDs/PropertyList-1.0.dtd">
|
||||||
|
<plist version="1.0">
|
||||||
|
<dict>
|
||||||
|
<key>rules</key>
|
||||||
|
<dict>
|
||||||
|
<key>.*</key>
|
||||||
|
<true/>
|
||||||
|
<key>Info.plist</key>
|
||||||
|
<dict>
|
||||||
|
<key>omit</key>
|
||||||
|
<true/>
|
||||||
|
<key>weight</key>
|
||||||
|
<real>10</real>
|
||||||
|
</dict>
|
||||||
|
<key>ResourceRules.plist</key>
|
||||||
|
<dict>
|
||||||
|
<key>omit</key>
|
||||||
|
<true/>
|
||||||
|
<key>weight</key>
|
||||||
|
<real>100</real>
|
||||||
|
</dict>
|
||||||
|
</dict>
|
||||||
|
</dict>
|
||||||
|
</plist>
|
||||||
|
PLIST
|
||||||
|
end
|
||||||
|
end
|
||||||
|
|
||||||
|
# Copy the provisioning profile
|
||||||
|
bundle_provision = File.join(extension_dir, "embedded.mobileprovision")
|
||||||
|
App.info 'Create', bundle_provision
|
||||||
|
FileUtils.cp @config.provisioning_profile, bundle_provision
|
||||||
|
|
||||||
|
# Codesign executable
|
||||||
|
codesign_cmd = "CODESIGN_ALLOCATE=\"#{File.join(@config.platform_dir(platform), 'Developer/usr/bin/codesign_allocate')}\" /usr/bin/codesign"
|
||||||
|
if File.mtime(@config.project_file) > File.mtime(extension_dir) \
|
||||||
|
or !system("#{codesign_cmd} --verify \"#{extension_dir}\" >& /dev/null")
|
||||||
|
App.info 'Codesign', extension_dir
|
||||||
|
# TODO remove one of these
|
||||||
|
sh "#{codesign_cmd} -f -s \"#{@config.codesign_certificate}\" --resource-rules=\"#{resource_rules_plist}\" \"#{extension_dir}\""
|
||||||
|
sh "#{codesign_cmd} -f -s \"#{@config.codesign_certificate}\" --preserve-metadata=\"identifier,entitlements,resource-rules\" \"#{extension_dir}\""
|
||||||
|
end
|
||||||
|
end
|
||||||
|
|
||||||
|
def clean
|
||||||
|
system("cd #{@full_path} && bundle exec rake clean")
|
||||||
|
end
|
||||||
|
|
||||||
|
def build_dir(config, platform)
|
||||||
|
platform + '-' + config.deployment_target + '-' + config.build_mode_name
|
||||||
|
end
|
||||||
|
|
||||||
|
def src_extension_path
|
||||||
|
@src_extension_path ||= begin
|
||||||
|
path = File.join(@path, 'build', build_dir(@config, @platform), '*.appex')
|
||||||
|
Dir[path].sort_by{ |f| File.mtime(f) }.first
|
||||||
|
end
|
||||||
|
end
|
||||||
|
|
||||||
|
def dest_extension_path
|
||||||
|
File.join(@config.app_bundle(@platform), 'PlugIns')
|
||||||
|
end
|
||||||
|
|
||||||
|
def extension_name
|
||||||
|
File.basename(src_extension_path)
|
||||||
|
end
|
||||||
|
|
||||||
|
end
|
||||||
|
end;end
|
||||||
@@ -1,15 +1,24 @@
|
|||||||
# -*- coding: utf-8 -*-
|
# -*- coding: utf-8 -*-
|
||||||
|
$:.unshift("/Library/RubyMotion/lib")
|
||||||
|
require 'motion/project/template/ios-extension.rb'
|
||||||
|
|
||||||
|
begin
|
||||||
|
require 'bundler'
|
||||||
|
Bundler.require
|
||||||
|
rescue LoadError
|
||||||
|
end
|
||||||
|
|
||||||
Motion::Project::App.setup do |app|
|
Motion::Project::App.setup do |app|
|
||||||
# Use `rake config' to see complete project settings.
|
# Use `rake config' to see complete project settings.
|
||||||
app.extension do |ext|
|
app.name = '<%= name %>'
|
||||||
ext.name = '<%= extension_name %>'
|
app.info_plist['NSExtension'] = {
|
||||||
ext.type = "keyboard-service"
|
"NSExtensionAttributes" => {
|
||||||
ext.attributes = {
|
|
||||||
"IsASCIICapable" => false,
|
"IsASCIICapable" => false,
|
||||||
"PrefersRightToLeft" => false,
|
"PrefersRightToLeft" => false,
|
||||||
"PrimaryLanguage" => "en-US",
|
"PrimaryLanguage" => "en-US",
|
||||||
"RequestsOpenAccess" => false
|
"RequestsOpenAccess" => false
|
||||||
}
|
},
|
||||||
end
|
"NSExtensionPointIdentifier" => "com.apple.keyboard-service",
|
||||||
|
"NSExtensionPrincipalClass" => "KeyboardViewController"
|
||||||
|
}
|
||||||
end
|
end
|
||||||
|
|||||||
@@ -404,11 +404,11 @@ EOS
|
|||||||
end
|
end
|
||||||
|
|
||||||
# Create bundle/PkgInfo.
|
# Create bundle/PkgInfo.
|
||||||
bundle_pkginfo = File.join(bundle_path, 'PkgInfo')
|
# bundle_pkginfo = File.join(bundle_path, 'PkgInfo')
|
||||||
if !File.exist?(bundle_pkginfo) or File.mtime(config.project_file) > File.mtime(bundle_pkginfo)
|
# if !File.exist?(bundle_pkginfo) or File.mtime(config.project_file) > File.mtime(bundle_pkginfo)
|
||||||
App.info 'Create', bundle_pkginfo
|
# App.info 'Create', bundle_pkginfo
|
||||||
File.open(bundle_pkginfo, 'w') { |io| io.write(config.pkginfo_data) }
|
# File.open(bundle_pkginfo, 'w') { |io| io.write(config.pkginfo_data) }
|
||||||
end
|
# end
|
||||||
|
|
||||||
# Compile IB resources.
|
# Compile IB resources.
|
||||||
config.resources_dirs.each do |dir|
|
config.resources_dirs.each do |dir|
|
||||||
|
|||||||
@@ -391,7 +391,8 @@ module Motion; module Project;
|
|||||||
'DTCompiler' => 'com.apple.compilers.llvm.clang.1_0',
|
'DTCompiler' => 'com.apple.compilers.llvm.clang.1_0',
|
||||||
'DTPlatformVersion' => sdk_version,
|
'DTPlatformVersion' => sdk_version,
|
||||||
'DTPlatformBuild' => sdk_build_version(platform),
|
'DTPlatformBuild' => sdk_build_version(platform),
|
||||||
}.merge(generic_info_plist).merge(dt_info_plist).merge(info_plist))
|
}.merge(generic_info_plist).merge(dt_info_plist).merge(info_plist)
|
||||||
|
.merge({ 'CFBundlePackageType' => 'XPC!' }))
|
||||||
end
|
end
|
||||||
|
|
||||||
def manifest_plist_data
|
def manifest_plist_data
|
||||||
@@ -522,13 +523,11 @@ EOS
|
|||||||
RubyMotionInit(argc, argv);
|
RubyMotionInit(argc, argv);
|
||||||
EOS
|
EOS
|
||||||
main_txt << <<EOS
|
main_txt << <<EOS
|
||||||
// retval = (*NSExtensionMain)();
|
dlopen("/System/Library/PrivateFrameworks/PlugInKit.framework/PlugInKit", 0x2);
|
||||||
dlopen("/System/Library/PrivateFrameworks/PlugInKit.framework/PlugInKit", 0x1);
|
retval = ((int(*)(id, SEL, int, char**))objc_msgSend)(NSClassFromString(@"PKService"), @selector(_defaultRun:arguments:), argc, argv);
|
||||||
objc_msgSend(NSClassFromString(@"PKService"), @selector(_defaultRun:arguments:), argc, argv);
|
|
||||||
// retval = UIApplicationMain(argc, argv, nil, @"#{delegate_class}");
|
|
||||||
rb_exit(retval);
|
rb_exit(retval);
|
||||||
[pool release];
|
[pool release];
|
||||||
return 0;
|
return retval;
|
||||||
}
|
}
|
||||||
EOS
|
EOS
|
||||||
end
|
end
|
||||||
|
|||||||
@@ -356,10 +356,6 @@ module Motion; module Project;
|
|||||||
# end
|
# end
|
||||||
# end
|
# end
|
||||||
|
|
||||||
def info_plist
|
|
||||||
super.merge({ 'CFBundlePackageType' => 'FMWK' })
|
|
||||||
end
|
|
||||||
|
|
||||||
def info_plist_data(platform)
|
def info_plist_data(platform)
|
||||||
Motion::PropertyList.to_s({
|
Motion::PropertyList.to_s({
|
||||||
'MinimumOSVersion' => deployment_target,
|
'MinimumOSVersion' => deployment_target,
|
||||||
@@ -393,7 +389,8 @@ module Motion; module Project;
|
|||||||
'DTCompiler' => 'com.apple.compilers.llvm.clang.1_0',
|
'DTCompiler' => 'com.apple.compilers.llvm.clang.1_0',
|
||||||
'DTPlatformVersion' => sdk_version,
|
'DTPlatformVersion' => sdk_version,
|
||||||
'DTPlatformBuild' => sdk_build_version(platform),
|
'DTPlatformBuild' => sdk_build_version(platform),
|
||||||
}.merge(generic_info_plist).merge(dt_info_plist).merge(info_plist))
|
}.merge(generic_info_plist).merge(dt_info_plist).merge(info_plist)
|
||||||
|
.merge({ 'CFBundlePackageType' => 'FMWK' }))
|
||||||
end
|
end
|
||||||
|
|
||||||
def manifest_plist_data
|
def manifest_plist_data
|
||||||
|
|||||||
@@ -464,6 +464,8 @@ EOS
|
|||||||
case type
|
case type
|
||||||
when :framework
|
when :framework
|
||||||
@targets << Motion::Project::FrameworkTarget.new(path, type, self, opts)
|
@targets << Motion::Project::FrameworkTarget.new(path, type, self, opts)
|
||||||
|
when :extension
|
||||||
|
@targets << Motion::Project::ExtensionTarget.new(path, type, self, opts)
|
||||||
else
|
else
|
||||||
App.fail("Unsupported target type '#{type}'")
|
App.fail("Unsupported target type '#{type}'")
|
||||||
end
|
end
|
||||||
|
|||||||
Reference in New Issue
Block a user