@adstage/react-native-sdk 1.0.6 → 1.0.7

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
package/app.plugin.js ADDED
@@ -0,0 +1,3 @@
1
+ // Expo Config Plugin entry point
2
+ // 빌드된 플러그인을 내보냅니다.
3
+ module.exports = require('./plugin/build/index');
package/package.json CHANGED
@@ -1,6 +1,6 @@
1
1
  {
2
2
  "name": "@adstage/react-native-sdk",
3
- "version": "1.0.6",
3
+ "version": "1.0.7",
4
4
  "description": "AdStage SDK for React Native - 광고 어트리뷰션, 딥링크, 인앱 이벤트 트래킹",
5
5
  "main": "lib/commonjs/index",
6
6
  "module": "lib/module/index",
@@ -12,6 +12,8 @@
12
12
  "lib",
13
13
  "android",
14
14
  "ios",
15
+ "plugin",
16
+ "app.plugin.js",
15
17
  "*.podspec",
16
18
  "!ios/build",
17
19
  "!android/build",
@@ -28,8 +30,9 @@
28
30
  "test": "jest",
29
31
  "typecheck": "tsc --noEmit",
30
32
  "lint": "eslint \"**/*.{js,ts,tsx}\"",
31
- "clean": "del-cli lib",
32
- "prepare": "bob build"
33
+ "clean": "del-cli lib plugin/build",
34
+ "build:plugin": "tsc -p plugin/tsconfig.json",
35
+ "prepare": "bob build && npm run build:plugin"
33
36
  },
34
37
  "keywords": [
35
38
  "react-native",
@@ -55,6 +58,7 @@
55
58
  "registry": "https://registry.npmjs.org/"
56
59
  },
57
60
  "devDependencies": {
61
+ "@expo/config-plugins": "^8.0.0",
58
62
  "@react-native/eslint-config": "^0.73.0",
59
63
  "@types/jest": "^29.5.0",
60
64
  "@types/react": "^18.2.0",
@@ -0,0 +1,16 @@
1
+ /**
2
+ * AdStage React Native SDK - Expo Config Plugin
3
+ *
4
+ * iOS AppDelegate에 딥링크 핸들링 코드를 자동으로 추가합니다.
5
+ *
6
+ * 사용법:
7
+ * app.json 또는 app.config.js에 추가:
8
+ * {
9
+ * "expo": {
10
+ * "plugins": ["@adstage/react-native-sdk"]
11
+ * }
12
+ * }
13
+ */
14
+ import { ConfigPlugin } from '@expo/config-plugins';
15
+ declare const _default: ConfigPlugin<void>;
16
+ export default _default;
@@ -0,0 +1,31 @@
1
+ "use strict";
2
+ /**
3
+ * AdStage React Native SDK - Expo Config Plugin
4
+ *
5
+ * iOS AppDelegate에 딥링크 핸들링 코드를 자동으로 추가합니다.
6
+ *
7
+ * 사용법:
8
+ * app.json 또는 app.config.js에 추가:
9
+ * {
10
+ * "expo": {
11
+ * "plugins": ["@adstage/react-native-sdk"]
12
+ * }
13
+ * }
14
+ */
15
+ Object.defineProperty(exports, "__esModule", { value: true });
16
+ const config_plugins_1 = require("@expo/config-plugins");
17
+ const withAdStageIOS_1 = require("./withAdStageIOS");
18
+ const pkg = require('../../package.json');
19
+ /**
20
+ * AdStage SDK Expo Config Plugin
21
+ *
22
+ * iOS에서 딥링크(URL Scheme, Universal Link)를 자동으로 처리하도록
23
+ * AppDelegate를 수정합니다.
24
+ */
25
+ const withAdStage = (config) => {
26
+ // iOS 설정 적용
27
+ config = (0, withAdStageIOS_1.withAdStageIOS)(config);
28
+ return config;
29
+ };
30
+ exports.default = (0, config_plugins_1.createRunOncePlugin)(withAdStage, pkg.name, pkg.version);
31
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiaW5kZXguanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi9zcmMvaW5kZXgudHMiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6IjtBQUFBOzs7Ozs7Ozs7Ozs7R0FZRzs7QUFFSCx5REFBeUU7QUFDekUscURBQWtEO0FBRWxELE1BQU0sR0FBRyxHQUFHLE9BQU8sQ0FBQyxvQkFBb0IsQ0FBQyxDQUFDO0FBRTFDOzs7OztHQUtHO0FBQ0gsTUFBTSxXQUFXLEdBQWlCLENBQUMsTUFBTSxFQUFFLEVBQUU7SUFDM0MsWUFBWTtJQUNaLE1BQU0sR0FBRyxJQUFBLCtCQUFjLEVBQUMsTUFBTSxDQUFDLENBQUM7SUFFaEMsT0FBTyxNQUFNLENBQUM7QUFDaEIsQ0FBQyxDQUFDO0FBRUYsa0JBQWUsSUFBQSxvQ0FBbUIsRUFBQyxXQUFXLEVBQUUsR0FBRyxDQUFDLElBQUksRUFBRSxHQUFHLENBQUMsT0FBTyxDQUFDLENBQUMiLCJzb3VyY2VzQ29udGVudCI6WyIvKipcbiAqIEFkU3RhZ2UgUmVhY3QgTmF0aXZlIFNESyAtIEV4cG8gQ29uZmlnIFBsdWdpblxuICpcbiAqIGlPUyBBcHBEZWxlZ2F0ZeyXkCDrlKXrp4Htgawg7ZW465Ok66eBIOy9lOuTnOulvCDsnpDrj5nsnLzroZwg7LaU6rCA7ZWp64uI64ukLlxuICpcbiAqIOyCrOyaqeuylTpcbiAqIGFwcC5qc29uIOuYkOuKlCBhcHAuY29uZmlnLmpz7JeQIOy2lOqwgDpcbiAqIHtcbiAqICAgXCJleHBvXCI6IHtcbiAqICAgICBcInBsdWdpbnNcIjogW1wiQGFkc3RhZ2UvcmVhY3QtbmF0aXZlLXNka1wiXVxuICogICB9XG4gKiB9XG4gKi9cblxuaW1wb3J0IHsgQ29uZmlnUGx1Z2luLCBjcmVhdGVSdW5PbmNlUGx1Z2luIH0gZnJvbSAnQGV4cG8vY29uZmlnLXBsdWdpbnMnO1xuaW1wb3J0IHsgd2l0aEFkU3RhZ2VJT1MgfSBmcm9tICcuL3dpdGhBZFN0YWdlSU9TJztcblxuY29uc3QgcGtnID0gcmVxdWlyZSgnLi4vLi4vcGFja2FnZS5qc29uJyk7XG5cbi8qKlxuICogQWRTdGFnZSBTREsgRXhwbyBDb25maWcgUGx1Z2luXG4gKlxuICogaU9T7JeQ7IScIOuUpeunge2BrChVUkwgU2NoZW1lLCBVbml2ZXJzYWwgTGluaynrpbwg7J6Q64+Z7Jy866GcIOyymOumrO2VmOuPhOuhnVxuICogQXBwRGVsZWdhdGXrpbwg7IiY7KCV7ZWp64uI64ukLlxuICovXG5jb25zdCB3aXRoQWRTdGFnZTogQ29uZmlnUGx1Z2luID0gKGNvbmZpZykgPT4ge1xuICAvLyBpT1Mg7ISk7KCVIOyggeyaqVxuICBjb25maWcgPSB3aXRoQWRTdGFnZUlPUyhjb25maWcpO1xuXG4gIHJldHVybiBjb25maWc7XG59O1xuXG5leHBvcnQgZGVmYXVsdCBjcmVhdGVSdW5PbmNlUGx1Z2luKHdpdGhBZFN0YWdlLCBwa2cubmFtZSwgcGtnLnZlcnNpb24pO1xuIl19
@@ -0,0 +1,11 @@
1
+ /**
2
+ * iOS AppDelegate 수정을 위한 Config Plugin
3
+ *
4
+ * AppDelegate.swift에 AdStage 딥링크 핸들링 코드를 추가합니다.
5
+ */
6
+ import { ConfigPlugin } from '@expo/config-plugins';
7
+ /**
8
+ * iOS AppDelegate에 AdStage 딥링크 핸들링 코드를 추가하는 Config Plugin
9
+ */
10
+ export declare const withAdStageIOS: ConfigPlugin;
11
+ export default withAdStageIOS;
@@ -0,0 +1,204 @@
1
+ "use strict";
2
+ /**
3
+ * iOS AppDelegate 수정을 위한 Config Plugin
4
+ *
5
+ * AppDelegate.swift에 AdStage 딥링크 핸들링 코드를 추가합니다.
6
+ */
7
+ Object.defineProperty(exports, "__esModule", { value: true });
8
+ exports.withAdStageIOS = void 0;
9
+ const config_plugins_1 = require("@expo/config-plugins");
10
+ /**
11
+ * AppDelegate.swift에 추가할 import 문
12
+ */
13
+ const ADSTAGE_IMPORT = 'import AdapterAdStage';
14
+ /**
15
+ * application:openURL:options: 메서드에 추가할 딥링크 핸들링 코드
16
+ */
17
+ const DEEPLINK_HANDLER_CODE = `
18
+ // AdStage 딥링크 처리
19
+ AdStageManager.shared.handleDeepLink(url)`;
20
+ /**
21
+ * application:continueUserActivity: 메서드에 추가할 Universal Link 핸들링 코드
22
+ */
23
+ const UNIVERSAL_LINK_HANDLER_CODE = `
24
+ // AdStage Universal Link 처리
25
+ if let url = userActivity.webpageURL {
26
+ AdStageManager.shared.handleUniversalLink(userActivity)
27
+ }`;
28
+ /**
29
+ * iOS AppDelegate에 AdStage 딥링크 핸들링 코드를 추가하는 Config Plugin
30
+ */
31
+ const withAdStageIOS = (config) => {
32
+ return (0, config_plugins_1.withAppDelegate)(config, (cfg) => {
33
+ const appDelegate = cfg.modResults;
34
+ if (appDelegate.language === 'swift') {
35
+ appDelegate.contents = modifySwiftAppDelegate(appDelegate.contents);
36
+ }
37
+ else if (appDelegate.language === 'objc' || appDelegate.language === 'objcpp') {
38
+ appDelegate.contents = modifyObjCAppDelegate(appDelegate.contents);
39
+ }
40
+ return cfg;
41
+ });
42
+ };
43
+ exports.withAdStageIOS = withAdStageIOS;
44
+ /**
45
+ * Swift AppDelegate 수정
46
+ */
47
+ function modifySwiftAppDelegate(contents) {
48
+ // 이미 AdStage import가 있는지 확인
49
+ if (contents.includes(ADSTAGE_IMPORT)) {
50
+ console.log('[AdStage] AppDelegate already configured, skipping...');
51
+ return contents;
52
+ }
53
+ // 1. import 문 추가
54
+ contents = addSwiftImport(contents, ADSTAGE_IMPORT);
55
+ // 2. application:openURL:options: 메서드에 딥링크 핸들러 추가
56
+ contents = addDeepLinkHandler(contents);
57
+ // 3. application:continueUserActivity: 메서드에 Universal Link 핸들러 추가
58
+ contents = addUniversalLinkHandler(contents);
59
+ return contents;
60
+ }
61
+ /**
62
+ * Swift import 문 추가
63
+ */
64
+ function addSwiftImport(contents, importStatement) {
65
+ // 마지막 import 문 뒤에 추가
66
+ const importRegex = /^import\s+\w+.*$/gm;
67
+ const matches = [...contents.matchAll(importRegex)];
68
+ if (matches.length > 0) {
69
+ const lastMatch = matches[matches.length - 1];
70
+ const insertPosition = lastMatch.index + lastMatch[0].length;
71
+ contents =
72
+ contents.slice(0, insertPosition) +
73
+ '\n' +
74
+ importStatement +
75
+ contents.slice(insertPosition);
76
+ }
77
+ return contents;
78
+ }
79
+ /**
80
+ * application:openURL:options: 메서드에 딥링크 핸들러 추가
81
+ */
82
+ function addDeepLinkHandler(contents) {
83
+ // application(_:open:options:) 메서드 찾기
84
+ const openURLMethodRegex = /func\s+application\s*\(\s*_\s+app:\s*UIApplication\s*,\s*open\s+url:\s*URL\s*,\s*options[^)]*\)\s*->\s*Bool\s*\{/;
85
+ const match = contents.match(openURLMethodRegex);
86
+ if (match) {
87
+ // 기존 메서드에 코드 추가
88
+ const insertPosition = match.index + match[0].length;
89
+ contents =
90
+ contents.slice(0, insertPosition) +
91
+ DEEPLINK_HANDLER_CODE +
92
+ contents.slice(insertPosition);
93
+ }
94
+ else {
95
+ // 메서드가 없으면 AppDelegate 클래스에 추가
96
+ contents = addOpenURLMethod(contents);
97
+ }
98
+ return contents;
99
+ }
100
+ /**
101
+ * application:openURL:options: 메서드 전체 추가
102
+ */
103
+ function addOpenURLMethod(contents) {
104
+ const newMethod = `
105
+
106
+ // MARK: - AdStage Deep Link Handling
107
+
108
+ func application(_ app: UIApplication, open url: URL, options: [UIApplication.OpenURLOptionsKey: Any] = [:]) -> Bool {
109
+ // AdStage 딥링크 처리
110
+ AdStageManager.shared.handleDeepLink(url)
111
+
112
+ // RCTLinkingManager로 전달 (React Native Linking 지원)
113
+ return RCTLinkingManager.application(app, open: url, options: options)
114
+ }`;
115
+ // 클래스 끝 (마지막 }) 앞에 삽입
116
+ const classEndRegex = /\n\s*\}\s*$/;
117
+ const match = contents.match(classEndRegex);
118
+ if (match) {
119
+ contents = contents.replace(classEndRegex, newMethod + match[0]);
120
+ }
121
+ return contents;
122
+ }
123
+ /**
124
+ * application:continueUserActivity: 메서드에 Universal Link 핸들러 추가
125
+ */
126
+ function addUniversalLinkHandler(contents) {
127
+ // application(_:continue:restorationHandler:) 메서드 찾기
128
+ const continueUserActivityRegex = /func\s+application\s*\([^)]*continue\s+userActivity:\s*NSUserActivity[^)]*\)\s*->\s*Bool\s*\{/;
129
+ const match = contents.match(continueUserActivityRegex);
130
+ if (match) {
131
+ // 기존 메서드에 코드 추가
132
+ const insertPosition = match.index + match[0].length;
133
+ contents =
134
+ contents.slice(0, insertPosition) +
135
+ UNIVERSAL_LINK_HANDLER_CODE +
136
+ contents.slice(insertPosition);
137
+ }
138
+ else {
139
+ // 메서드가 없으면 AppDelegate 클래스에 추가
140
+ contents = addContinueUserActivityMethod(contents);
141
+ }
142
+ return contents;
143
+ }
144
+ /**
145
+ * application:continueUserActivity: 메서드 전체 추가
146
+ */
147
+ function addContinueUserActivityMethod(contents) {
148
+ const newMethod = `
149
+
150
+ // MARK: - AdStage Universal Link Handling
151
+
152
+ func application(_ application: UIApplication, continue userActivity: NSUserActivity, restorationHandler: @escaping ([UIUserActivityRestoring]?) -> Void) -> Bool {
153
+ // AdStage Universal Link 처리
154
+ if let url = userActivity.webpageURL {
155
+ AdStageManager.shared.handleUniversalLink(userActivity)
156
+ }
157
+
158
+ // RCTLinkingManager로 전달 (React Native Linking 지원)
159
+ return RCTLinkingManager.application(application, continue: userActivity, restorationHandler: restorationHandler)
160
+ }`;
161
+ // 클래스 끝 (마지막 }) 앞에 삽입
162
+ const classEndRegex = /\n\s*\}\s*$/;
163
+ const match = contents.match(classEndRegex);
164
+ if (match) {
165
+ contents = contents.replace(classEndRegex, newMethod + match[0]);
166
+ }
167
+ return contents;
168
+ }
169
+ /**
170
+ * Objective-C AppDelegate 수정 (레거시 지원)
171
+ */
172
+ function modifyObjCAppDelegate(contents) {
173
+ // 이미 AdStage import가 있는지 확인
174
+ if (contents.includes('#import <AdapterAdStage/AdapterAdStage.h>')) {
175
+ console.log('[AdStage] AppDelegate already configured, skipping...');
176
+ return contents;
177
+ }
178
+ // 1. import 문 추가
179
+ const importRegex = /#import\s+[<"][^>"]+[>"]/g;
180
+ const matches = [...contents.matchAll(importRegex)];
181
+ if (matches.length > 0) {
182
+ const lastMatch = matches[matches.length - 1];
183
+ const insertPosition = lastMatch.index + lastMatch[0].length;
184
+ contents =
185
+ contents.slice(0, insertPosition) +
186
+ '\n#import <AdapterAdStage/AdapterAdStage.h>' +
187
+ contents.slice(insertPosition);
188
+ }
189
+ // 2. openURL 메서드에 딥링크 핸들러 추가
190
+ const openURLMethodRegex = /-\s*\(BOOL\)\s*application:\s*\(UIApplication\s*\*\)\s*\w+\s+openURL:\s*\(NSURL\s*\*\)\s*(\w+)/;
191
+ const openURLMatch = contents.match(openURLMethodRegex);
192
+ if (openURLMatch) {
193
+ const urlVarName = openURLMatch[1];
194
+ const methodBodyStart = contents.indexOf('{', openURLMatch.index) + 1;
195
+ const handlerCode = `\n // AdStage 딥링크 처리\n [[AdStageManager shared] handleDeepLink:${urlVarName}];\n`;
196
+ contents =
197
+ contents.slice(0, methodBodyStart) +
198
+ handlerCode +
199
+ contents.slice(methodBodyStart);
200
+ }
201
+ return contents;
202
+ }
203
+ exports.default = exports.withAdStageIOS;
204
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"withAdStageIOS.js","sourceRoot":"","sources":["../src/withAdStageIOS.ts"],"names":[],"mappings":"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","sourcesContent":["/**\n * iOS AppDelegate 수정을 위한 Config Plugin\n *\n * AppDelegate.swift에 AdStage 딥링크 핸들링 코드를 추가합니다.\n */\n\nimport { ConfigPlugin, withAppDelegate } from '@expo/config-plugins';\n\n/**\n * AppDelegate.swift에 추가할 import 문\n */\nconst ADSTAGE_IMPORT = 'import AdapterAdStage';\n\n/**\n * application:openURL:options: 메서드에 추가할 딥링크 핸들링 코드\n */\nconst DEEPLINK_HANDLER_CODE = `\n    // AdStage 딥링크 처리\n    AdStageManager.shared.handleDeepLink(url)`;\n\n/**\n * application:continueUserActivity: 메서드에 추가할 Universal Link 핸들링 코드\n */\nconst UNIVERSAL_LINK_HANDLER_CODE = `\n    // AdStage Universal Link 처리\n    if let url = userActivity.webpageURL {\n      AdStageManager.shared.handleUniversalLink(userActivity)\n    }`;\n\n/**\n * iOS AppDelegate에 AdStage 딥링크 핸들링 코드를 추가하는 Config Plugin\n */\nexport const withAdStageIOS: ConfigPlugin = (config) => {\n  return withAppDelegate(config, (cfg) => {\n    const appDelegate = cfg.modResults;\n\n    if (appDelegate.language === 'swift') {\n      appDelegate.contents = modifySwiftAppDelegate(appDelegate.contents);\n    } else if (appDelegate.language === 'objc' || appDelegate.language === 'objcpp') {\n      appDelegate.contents = modifyObjCAppDelegate(appDelegate.contents);\n    }\n\n    return cfg;\n  });\n};\n\n/**\n * Swift AppDelegate 수정\n */\nfunction modifySwiftAppDelegate(contents: string): string {\n  // 이미 AdStage import가 있는지 확인\n  if (contents.includes(ADSTAGE_IMPORT)) {\n    console.log('[AdStage] AppDelegate already configured, skipping...');\n    return contents;\n  }\n\n  // 1. import 문 추가\n  contents = addSwiftImport(contents, ADSTAGE_IMPORT);\n\n  // 2. application:openURL:options: 메서드에 딥링크 핸들러 추가\n  contents = addDeepLinkHandler(contents);\n\n  // 3. application:continueUserActivity: 메서드에 Universal Link 핸들러 추가\n  contents = addUniversalLinkHandler(contents);\n\n  return contents;\n}\n\n/**\n * Swift import 문 추가\n */\nfunction addSwiftImport(contents: string, importStatement: string): string {\n  // 마지막 import 문 뒤에 추가\n  const importRegex = /^import\\s+\\w+.*$/gm;\n  const matches = [...contents.matchAll(importRegex)];\n\n  if (matches.length > 0) {\n    const lastMatch = matches[matches.length - 1];\n    const insertPosition = lastMatch.index! + lastMatch[0].length;\n    contents =\n      contents.slice(0, insertPosition) +\n      '\\n' +\n      importStatement +\n      contents.slice(insertPosition);\n  }\n\n  return contents;\n}\n\n/**\n * application:openURL:options: 메서드에 딥링크 핸들러 추가\n */\nfunction addDeepLinkHandler(contents: string): string {\n  // application(_:open:options:) 메서드 찾기\n  const openURLMethodRegex =\n    /func\\s+application\\s*\\(\\s*_\\s+app:\\s*UIApplication\\s*,\\s*open\\s+url:\\s*URL\\s*,\\s*options[^)]*\\)\\s*->\\s*Bool\\s*\\{/;\n\n  const match = contents.match(openURLMethodRegex);\n\n  if (match) {\n    // 기존 메서드에 코드 추가\n    const insertPosition = match.index! + match[0].length;\n    contents =\n      contents.slice(0, insertPosition) +\n      DEEPLINK_HANDLER_CODE +\n      contents.slice(insertPosition);\n  } else {\n    // 메서드가 없으면 AppDelegate 클래스에 추가\n    contents = addOpenURLMethod(contents);\n  }\n\n  return contents;\n}\n\n/**\n * application:openURL:options: 메서드 전체 추가\n */\nfunction addOpenURLMethod(contents: string): string {\n  const newMethod = `\n\n  // MARK: - AdStage Deep Link Handling\n  \n  func application(_ app: UIApplication, open url: URL, options: [UIApplication.OpenURLOptionsKey: Any] = [:]) -> Bool {\n    // AdStage 딥링크 처리\n    AdStageManager.shared.handleDeepLink(url)\n    \n    // RCTLinkingManager로 전달 (React Native Linking 지원)\n    return RCTLinkingManager.application(app, open: url, options: options)\n  }`;\n\n  // 클래스 끝 (마지막 }) 앞에 삽입\n  const classEndRegex = /\\n\\s*\\}\\s*$/;\n  const match = contents.match(classEndRegex);\n\n  if (match) {\n    contents = contents.replace(classEndRegex, newMethod + match[0]);\n  }\n\n  return contents;\n}\n\n/**\n * application:continueUserActivity: 메서드에 Universal Link 핸들러 추가\n */\nfunction addUniversalLinkHandler(contents: string): string {\n  // application(_:continue:restorationHandler:) 메서드 찾기\n  const continueUserActivityRegex =\n    /func\\s+application\\s*\\([^)]*continue\\s+userActivity:\\s*NSUserActivity[^)]*\\)\\s*->\\s*Bool\\s*\\{/;\n\n  const match = contents.match(continueUserActivityRegex);\n\n  if (match) {\n    // 기존 메서드에 코드 추가\n    const insertPosition = match.index! + match[0].length;\n    contents =\n      contents.slice(0, insertPosition) +\n      UNIVERSAL_LINK_HANDLER_CODE +\n      contents.slice(insertPosition);\n  } else {\n    // 메서드가 없으면 AppDelegate 클래스에 추가\n    contents = addContinueUserActivityMethod(contents);\n  }\n\n  return contents;\n}\n\n/**\n * application:continueUserActivity: 메서드 전체 추가\n */\nfunction addContinueUserActivityMethod(contents: string): string {\n  const newMethod = `\n\n  // MARK: - AdStage Universal Link Handling\n  \n  func application(_ application: UIApplication, continue userActivity: NSUserActivity, restorationHandler: @escaping ([UIUserActivityRestoring]?) -> Void) -> Bool {\n    // AdStage Universal Link 처리\n    if let url = userActivity.webpageURL {\n      AdStageManager.shared.handleUniversalLink(userActivity)\n    }\n    \n    // RCTLinkingManager로 전달 (React Native Linking 지원)\n    return RCTLinkingManager.application(application, continue: userActivity, restorationHandler: restorationHandler)\n  }`;\n\n  // 클래스 끝 (마지막 }) 앞에 삽입\n  const classEndRegex = /\\n\\s*\\}\\s*$/;\n  const match = contents.match(classEndRegex);\n\n  if (match) {\n    contents = contents.replace(classEndRegex, newMethod + match[0]);\n  }\n\n  return contents;\n}\n\n/**\n * Objective-C AppDelegate 수정 (레거시 지원)\n */\nfunction modifyObjCAppDelegate(contents: string): string {\n  // 이미 AdStage import가 있는지 확인\n  if (contents.includes('#import <AdapterAdStage/AdapterAdStage.h>')) {\n    console.log('[AdStage] AppDelegate already configured, skipping...');\n    return contents;\n  }\n\n  // 1. import 문 추가\n  const importRegex = /#import\\s+[<\"][^>\"]+[>\"]/g;\n  const matches = [...contents.matchAll(importRegex)];\n\n  if (matches.length > 0) {\n    const lastMatch = matches[matches.length - 1];\n    const insertPosition = lastMatch.index! + lastMatch[0].length;\n    contents =\n      contents.slice(0, insertPosition) +\n      '\\n#import <AdapterAdStage/AdapterAdStage.h>' +\n      contents.slice(insertPosition);\n  }\n\n  // 2. openURL 메서드에 딥링크 핸들러 추가\n  const openURLMethodRegex =\n    /-\\s*\\(BOOL\\)\\s*application:\\s*\\(UIApplication\\s*\\*\\)\\s*\\w+\\s+openURL:\\s*\\(NSURL\\s*\\*\\)\\s*(\\w+)/;\n\n  const openURLMatch = contents.match(openURLMethodRegex);\n  if (openURLMatch) {\n    const urlVarName = openURLMatch[1];\n    const methodBodyStart = contents.indexOf('{', openURLMatch.index!) + 1;\n    const handlerCode = `\\n  // AdStage 딥링크 처리\\n  [[AdStageManager shared] handleDeepLink:${urlVarName}];\\n`;\n    contents =\n      contents.slice(0, methodBodyStart) +\n      handlerCode +\n      contents.slice(methodBodyStart);\n  }\n\n  return contents;\n}\n\nexport default withAdStageIOS;\n"]}
@@ -0,0 +1,33 @@
1
+ /**
2
+ * AdStage React Native SDK - Expo Config Plugin
3
+ *
4
+ * iOS AppDelegate에 딥링크 핸들링 코드를 자동으로 추가합니다.
5
+ *
6
+ * 사용법:
7
+ * app.json 또는 app.config.js에 추가:
8
+ * {
9
+ * "expo": {
10
+ * "plugins": ["@adstage/react-native-sdk"]
11
+ * }
12
+ * }
13
+ */
14
+
15
+ import { ConfigPlugin, createRunOncePlugin } from '@expo/config-plugins';
16
+ import { withAdStageIOS } from './withAdStageIOS';
17
+
18
+ const pkg = require('../../package.json');
19
+
20
+ /**
21
+ * AdStage SDK Expo Config Plugin
22
+ *
23
+ * iOS에서 딥링크(URL Scheme, Universal Link)를 자동으로 처리하도록
24
+ * AppDelegate를 수정합니다.
25
+ */
26
+ const withAdStage: ConfigPlugin = (config) => {
27
+ // iOS 설정 적용
28
+ config = withAdStageIOS(config);
29
+
30
+ return config;
31
+ };
32
+
33
+ export default createRunOncePlugin(withAdStage, pkg.name, pkg.version);
@@ -0,0 +1,237 @@
1
+ /**
2
+ * iOS AppDelegate 수정을 위한 Config Plugin
3
+ *
4
+ * AppDelegate.swift에 AdStage 딥링크 핸들링 코드를 추가합니다.
5
+ */
6
+
7
+ import { ConfigPlugin, withAppDelegate } from '@expo/config-plugins';
8
+
9
+ /**
10
+ * AppDelegate.swift에 추가할 import 문
11
+ */
12
+ const ADSTAGE_IMPORT = 'import AdapterAdStage';
13
+
14
+ /**
15
+ * application:openURL:options: 메서드에 추가할 딥링크 핸들링 코드
16
+ */
17
+ const DEEPLINK_HANDLER_CODE = `
18
+ // AdStage 딥링크 처리
19
+ AdStageManager.shared.handleDeepLink(url)`;
20
+
21
+ /**
22
+ * application:continueUserActivity: 메서드에 추가할 Universal Link 핸들링 코드
23
+ */
24
+ const UNIVERSAL_LINK_HANDLER_CODE = `
25
+ // AdStage Universal Link 처리
26
+ if let url = userActivity.webpageURL {
27
+ AdStageManager.shared.handleUniversalLink(userActivity)
28
+ }`;
29
+
30
+ /**
31
+ * iOS AppDelegate에 AdStage 딥링크 핸들링 코드를 추가하는 Config Plugin
32
+ */
33
+ export const withAdStageIOS: ConfigPlugin = (config) => {
34
+ return withAppDelegate(config, (cfg) => {
35
+ const appDelegate = cfg.modResults;
36
+
37
+ if (appDelegate.language === 'swift') {
38
+ appDelegate.contents = modifySwiftAppDelegate(appDelegate.contents);
39
+ } else if (appDelegate.language === 'objc' || appDelegate.language === 'objcpp') {
40
+ appDelegate.contents = modifyObjCAppDelegate(appDelegate.contents);
41
+ }
42
+
43
+ return cfg;
44
+ });
45
+ };
46
+
47
+ /**
48
+ * Swift AppDelegate 수정
49
+ */
50
+ function modifySwiftAppDelegate(contents: string): string {
51
+ // 이미 AdStage import가 있는지 확인
52
+ if (contents.includes(ADSTAGE_IMPORT)) {
53
+ console.log('[AdStage] AppDelegate already configured, skipping...');
54
+ return contents;
55
+ }
56
+
57
+ // 1. import 문 추가
58
+ contents = addSwiftImport(contents, ADSTAGE_IMPORT);
59
+
60
+ // 2. application:openURL:options: 메서드에 딥링크 핸들러 추가
61
+ contents = addDeepLinkHandler(contents);
62
+
63
+ // 3. application:continueUserActivity: 메서드에 Universal Link 핸들러 추가
64
+ contents = addUniversalLinkHandler(contents);
65
+
66
+ return contents;
67
+ }
68
+
69
+ /**
70
+ * Swift import 문 추가
71
+ */
72
+ function addSwiftImport(contents: string, importStatement: string): string {
73
+ // 마지막 import 문 뒤에 추가
74
+ const importRegex = /^import\s+\w+.*$/gm;
75
+ const matches = [...contents.matchAll(importRegex)];
76
+
77
+ if (matches.length > 0) {
78
+ const lastMatch = matches[matches.length - 1];
79
+ const insertPosition = lastMatch.index! + lastMatch[0].length;
80
+ contents =
81
+ contents.slice(0, insertPosition) +
82
+ '\n' +
83
+ importStatement +
84
+ contents.slice(insertPosition);
85
+ }
86
+
87
+ return contents;
88
+ }
89
+
90
+ /**
91
+ * application:openURL:options: 메서드에 딥링크 핸들러 추가
92
+ */
93
+ function addDeepLinkHandler(contents: string): string {
94
+ // application(_:open:options:) 메서드 찾기
95
+ const openURLMethodRegex =
96
+ /func\s+application\s*\(\s*_\s+app:\s*UIApplication\s*,\s*open\s+url:\s*URL\s*,\s*options[^)]*\)\s*->\s*Bool\s*\{/;
97
+
98
+ const match = contents.match(openURLMethodRegex);
99
+
100
+ if (match) {
101
+ // 기존 메서드에 코드 추가
102
+ const insertPosition = match.index! + match[0].length;
103
+ contents =
104
+ contents.slice(0, insertPosition) +
105
+ DEEPLINK_HANDLER_CODE +
106
+ contents.slice(insertPosition);
107
+ } else {
108
+ // 메서드가 없으면 AppDelegate 클래스에 추가
109
+ contents = addOpenURLMethod(contents);
110
+ }
111
+
112
+ return contents;
113
+ }
114
+
115
+ /**
116
+ * application:openURL:options: 메서드 전체 추가
117
+ */
118
+ function addOpenURLMethod(contents: string): string {
119
+ const newMethod = `
120
+
121
+ // MARK: - AdStage Deep Link Handling
122
+
123
+ func application(_ app: UIApplication, open url: URL, options: [UIApplication.OpenURLOptionsKey: Any] = [:]) -> Bool {
124
+ // AdStage 딥링크 처리
125
+ AdStageManager.shared.handleDeepLink(url)
126
+
127
+ // RCTLinkingManager로 전달 (React Native Linking 지원)
128
+ return RCTLinkingManager.application(app, open: url, options: options)
129
+ }`;
130
+
131
+ // 클래스 끝 (마지막 }) 앞에 삽입
132
+ const classEndRegex = /\n\s*\}\s*$/;
133
+ const match = contents.match(classEndRegex);
134
+
135
+ if (match) {
136
+ contents = contents.replace(classEndRegex, newMethod + match[0]);
137
+ }
138
+
139
+ return contents;
140
+ }
141
+
142
+ /**
143
+ * application:continueUserActivity: 메서드에 Universal Link 핸들러 추가
144
+ */
145
+ function addUniversalLinkHandler(contents: string): string {
146
+ // application(_:continue:restorationHandler:) 메서드 찾기
147
+ const continueUserActivityRegex =
148
+ /func\s+application\s*\([^)]*continue\s+userActivity:\s*NSUserActivity[^)]*\)\s*->\s*Bool\s*\{/;
149
+
150
+ const match = contents.match(continueUserActivityRegex);
151
+
152
+ if (match) {
153
+ // 기존 메서드에 코드 추가
154
+ const insertPosition = match.index! + match[0].length;
155
+ contents =
156
+ contents.slice(0, insertPosition) +
157
+ UNIVERSAL_LINK_HANDLER_CODE +
158
+ contents.slice(insertPosition);
159
+ } else {
160
+ // 메서드가 없으면 AppDelegate 클래스에 추가
161
+ contents = addContinueUserActivityMethod(contents);
162
+ }
163
+
164
+ return contents;
165
+ }
166
+
167
+ /**
168
+ * application:continueUserActivity: 메서드 전체 추가
169
+ */
170
+ function addContinueUserActivityMethod(contents: string): string {
171
+ const newMethod = `
172
+
173
+ // MARK: - AdStage Universal Link Handling
174
+
175
+ func application(_ application: UIApplication, continue userActivity: NSUserActivity, restorationHandler: @escaping ([UIUserActivityRestoring]?) -> Void) -> Bool {
176
+ // AdStage Universal Link 처리
177
+ if let url = userActivity.webpageURL {
178
+ AdStageManager.shared.handleUniversalLink(userActivity)
179
+ }
180
+
181
+ // RCTLinkingManager로 전달 (React Native Linking 지원)
182
+ return RCTLinkingManager.application(application, continue: userActivity, restorationHandler: restorationHandler)
183
+ }`;
184
+
185
+ // 클래스 끝 (마지막 }) 앞에 삽입
186
+ const classEndRegex = /\n\s*\}\s*$/;
187
+ const match = contents.match(classEndRegex);
188
+
189
+ if (match) {
190
+ contents = contents.replace(classEndRegex, newMethod + match[0]);
191
+ }
192
+
193
+ return contents;
194
+ }
195
+
196
+ /**
197
+ * Objective-C AppDelegate 수정 (레거시 지원)
198
+ */
199
+ function modifyObjCAppDelegate(contents: string): string {
200
+ // 이미 AdStage import가 있는지 확인
201
+ if (contents.includes('#import <AdapterAdStage/AdapterAdStage.h>')) {
202
+ console.log('[AdStage] AppDelegate already configured, skipping...');
203
+ return contents;
204
+ }
205
+
206
+ // 1. import 문 추가
207
+ const importRegex = /#import\s+[<"][^>"]+[>"]/g;
208
+ const matches = [...contents.matchAll(importRegex)];
209
+
210
+ if (matches.length > 0) {
211
+ const lastMatch = matches[matches.length - 1];
212
+ const insertPosition = lastMatch.index! + lastMatch[0].length;
213
+ contents =
214
+ contents.slice(0, insertPosition) +
215
+ '\n#import <AdapterAdStage/AdapterAdStage.h>' +
216
+ contents.slice(insertPosition);
217
+ }
218
+
219
+ // 2. openURL 메서드에 딥링크 핸들러 추가
220
+ const openURLMethodRegex =
221
+ /-\s*\(BOOL\)\s*application:\s*\(UIApplication\s*\*\)\s*\w+\s+openURL:\s*\(NSURL\s*\*\)\s*(\w+)/;
222
+
223
+ const openURLMatch = contents.match(openURLMethodRegex);
224
+ if (openURLMatch) {
225
+ const urlVarName = openURLMatch[1];
226
+ const methodBodyStart = contents.indexOf('{', openURLMatch.index!) + 1;
227
+ const handlerCode = `\n // AdStage 딥링크 처리\n [[AdStageManager shared] handleDeepLink:${urlVarName}];\n`;
228
+ contents =
229
+ contents.slice(0, methodBodyStart) +
230
+ handlerCode +
231
+ contents.slice(methodBodyStart);
232
+ }
233
+
234
+ return contents;
235
+ }
236
+
237
+ export default withAdStageIOS;
@@ -0,0 +1,28 @@
1
+ {
2
+ "compilerOptions": {
3
+ "target": "ES2020",
4
+ "module": "commonjs",
5
+ "lib": ["ES2020"],
6
+ "declaration": true,
7
+ "strict": true,
8
+ "noImplicitAny": true,
9
+ "strictNullChecks": true,
10
+ "noImplicitThis": true,
11
+ "alwaysStrict": true,
12
+ "noUnusedLocals": false,
13
+ "noUnusedParameters": false,
14
+ "noImplicitReturns": true,
15
+ "noFallthroughCasesInSwitch": false,
16
+ "inlineSourceMap": true,
17
+ "inlineSources": true,
18
+ "experimentalDecorators": true,
19
+ "strictPropertyInitialization": false,
20
+ "outDir": "./build",
21
+ "rootDir": "./src",
22
+ "skipLibCheck": true,
23
+ "esModuleInterop": true,
24
+ "resolveJsonModule": true
25
+ },
26
+ "include": ["src/**/*"],
27
+ "exclude": ["node_modules", "build"]
28
+ }