Compare commits
50 Commits
443d8d0727
...
2.2.3
Author | SHA1 | Date | |
---|---|---|---|
f1ca8e92b7 | |||
|
436bc4b2af | ||
e0e561ff7f | |||
ed4e9849e7 | |||
ddf1d2509f | |||
2ae06867c7 | |||
f329386ccf | |||
5ad219ae89 | |||
beca2c6b2b | |||
3092376b65 | |||
c3b8ebfb37 | |||
166026a766 | |||
ccda606af5 | |||
756de4f1de | |||
8442c89944 | |||
57cedd37bb | |||
09556ba6e0 | |||
dea57dc1e2 | |||
07575bd2bf | |||
8686ae974c | |||
be4c561ea8 | |||
2357a40fff | |||
d4afa9c9e9 | |||
76ef0a2d59 | |||
129eb135f1 | |||
4ad15fcded | |||
fb2ddb2227 | |||
27f86f5c4d | |||
209ba49e3f | |||
ba07005b13 | |||
6c3f3a8982 | |||
0d651b810f | |||
1d7fc76340 | |||
5d4e461933 | |||
55264d61ad | |||
d21ad9d1ea | |||
0bd6c3c2d4 | |||
eed20367b9 | |||
43b5111d79 | |||
2983093a9c | |||
b4bbaa3bfd | |||
498c8fa4ae | |||
2957da6233 | |||
d79af06c38 | |||
d8937f2de6 | |||
9b27f24197 | |||
1d58fd5510 | |||
f6c49bf626 | |||
f1b62d83c4 | |||
ee5055efa5 |
317
.swiftlint.yml
@@ -1,43 +1,276 @@
|
|||||||
disabled_rules: # rule identifiers to exclude from running
|
# All rules here : https://realm.github.io/SwiftLint/rule-directory.html
|
||||||
- leading_whitespace
|
|
||||||
- trailing_whitespace
|
analyzer_rules:
|
||||||
- identifier_name
|
- capture_variable
|
||||||
- large_tuple
|
- typesafe_array_init
|
||||||
- file_length
|
- unused_declaration
|
||||||
- line_length
|
- unused_import
|
||||||
- force_try
|
|
||||||
- shorthand_operator
|
included:
|
||||||
- type_body_length
|
- Sources
|
||||||
- function_body_length
|
|
||||||
- function_parameter_count
|
## Rules configuration
|
||||||
- redundant_string_enum_value
|
|
||||||
- unused_closure_parameter
|
attributes:
|
||||||
- cyclomatic_complexity
|
always_on_line_above: ["@InjectedValue", "@ViewBuilder", "@IBOutlet"]
|
||||||
- syntactic_sugar
|
always_on_same_line: ["@Environment", "@EnvironmentObject", "@StateObject", "@State"]
|
||||||
- empty_enum_arguments
|
|
||||||
- force_cast
|
identifier_name:
|
||||||
- multiple_closures_with_trailing_closure
|
min_length:
|
||||||
- private_over_fileprivate
|
- 2
|
||||||
- trailing_comma
|
max_length:
|
||||||
- comment_spacing
|
- 60
|
||||||
excluded: # paths to ignore during linting. Takes precedence over `included`.
|
excluded:
|
||||||
- DerivedData
|
- x
|
||||||
- Carthage
|
- y
|
||||||
- Pods
|
|
||||||
- vendor
|
type_name:
|
||||||
- Vendor
|
min_length: 3
|
||||||
- "*/R2Tag+tags.swift"
|
max_length: 60
|
||||||
type_name:
|
excluded:
|
||||||
min_length: 1 # only warning
|
- T
|
||||||
max_length: # warning and error
|
allowed_symbols:
|
||||||
warning: 50
|
- _
|
||||||
error: 60
|
|
||||||
allowed_symbols: ["_"]
|
# line_length:
|
||||||
nesting:
|
# warning: 150
|
||||||
type_level:
|
|
||||||
warning: 3
|
disabled_rules:
|
||||||
error: 6
|
- blanket_disable_command # do not warn when rule is not re-enable later in the file
|
||||||
statement_level:
|
- type_contents_order
|
||||||
warning: 5
|
- legacy_objc_type
|
||||||
error: 10
|
- indentation_width
|
||||||
|
- function_parameter_count
|
||||||
|
- line_length
|
||||||
|
- function_body_length
|
||||||
|
- cyclomatic_complexity
|
||||||
|
- optional_data_string_conversion
|
||||||
|
|
||||||
|
opt_in_rules:
|
||||||
|
# Default rules :
|
||||||
|
- block_based_kvo
|
||||||
|
- class_delegate_protocol
|
||||||
|
- closing_brace
|
||||||
|
- closure_parameter_position
|
||||||
|
- colon
|
||||||
|
- comma
|
||||||
|
- comment_spacing
|
||||||
|
- compiler_protocol_init
|
||||||
|
- computed_accessors_order
|
||||||
|
- control_statement
|
||||||
|
- custom_rules
|
||||||
|
# - cyclomatic_complexity
|
||||||
|
- deployment_target
|
||||||
|
- discouraged_direct_init
|
||||||
|
- duplicate_enum_cases
|
||||||
|
- duplicate_imports
|
||||||
|
- duplicated_key_in_dictionary_literal
|
||||||
|
- dynamic_inline
|
||||||
|
- empty_enum_arguments
|
||||||
|
- empty_parameters
|
||||||
|
- empty_parentheses_with_trailing_closure
|
||||||
|
- file_length
|
||||||
|
- for_where
|
||||||
|
- force_unwrapping
|
||||||
|
- force_cast
|
||||||
|
- force_try
|
||||||
|
- trailing_whitespace
|
||||||
|
# - function_body_length
|
||||||
|
# - function_parameter_count
|
||||||
|
- generic_type_name
|
||||||
|
- identifier_name
|
||||||
|
- implicit_getter
|
||||||
|
- inclusive_language
|
||||||
|
- is_disjoint
|
||||||
|
- large_tuple
|
||||||
|
- leading_whitespace
|
||||||
|
- legacy_cggeometry_functions
|
||||||
|
- legacy_constant
|
||||||
|
- legacy_constructor
|
||||||
|
- legacy_hashing
|
||||||
|
- legacy_nsgeometry_functions
|
||||||
|
- legacy_random
|
||||||
|
# - line_length
|
||||||
|
- mark
|
||||||
|
- multiple_closures_with_trailing_closure
|
||||||
|
- nesting
|
||||||
|
- no_fallthrough_only
|
||||||
|
- no_space_in_method_call
|
||||||
|
- notification_center_detachment
|
||||||
|
- ns_number_init_as_function_reference
|
||||||
|
- nsobject_prefer_isequal
|
||||||
|
- opening_brace
|
||||||
|
- operator_whitespace
|
||||||
|
- orphaned_doc_comment
|
||||||
|
- private_over_fileprivate
|
||||||
|
- private_unit_test
|
||||||
|
- protocol_property_accessors_order
|
||||||
|
- reduce_boolean
|
||||||
|
- redundant_discardable_let
|
||||||
|
- redundant_objc_attribute
|
||||||
|
- redundant_optional_initialization
|
||||||
|
- redundant_set_access_control
|
||||||
|
- redundant_string_enum_value
|
||||||
|
- redundant_void_return
|
||||||
|
- return_arrow_whitespace
|
||||||
|
- self_in_property_initialization
|
||||||
|
- shorthand_operator
|
||||||
|
- statement_position
|
||||||
|
- superfluous_disable_command
|
||||||
|
- switch_case_alignment
|
||||||
|
- syntactic_sugar
|
||||||
|
- todo
|
||||||
|
- trailing_comma
|
||||||
|
- trailing_newline
|
||||||
|
- trailing_semicolon
|
||||||
|
- type_body_length
|
||||||
|
- type_name
|
||||||
|
- unavailable_condition
|
||||||
|
- unneeded_break_in_switch
|
||||||
|
- unused_closure_parameter
|
||||||
|
- unused_control_flow_label
|
||||||
|
- unused_enumerated
|
||||||
|
- unused_optional_binding
|
||||||
|
- unused_setter_value
|
||||||
|
- valid_ibinspectable
|
||||||
|
- vertical_parameter_alignment
|
||||||
|
- vertical_whitespace
|
||||||
|
- void_function_in_ternary
|
||||||
|
- void_return
|
||||||
|
- xctfail_message
|
||||||
|
- accessibility_trait_for_button
|
||||||
|
- array_init
|
||||||
|
- attributes
|
||||||
|
- closure_body_length
|
||||||
|
- closure_end_indentation
|
||||||
|
- closure_spacing
|
||||||
|
- collection_alignment
|
||||||
|
- comma_inheritance
|
||||||
|
- contains_over_filter_count
|
||||||
|
- contains_over_filter_is_empty
|
||||||
|
- contains_over_first_not_nil
|
||||||
|
- contains_over_range_nil_comparison
|
||||||
|
- convenience_type
|
||||||
|
- discarded_notification_center_observer
|
||||||
|
- discouraged_assert
|
||||||
|
- empty_count
|
||||||
|
- empty_string
|
||||||
|
- empty_xctest_method
|
||||||
|
- enum_case_associated_values_count
|
||||||
|
- explicit_init
|
||||||
|
- fallthrough
|
||||||
|
- fatal_error_message
|
||||||
|
- file_header
|
||||||
|
- first_where
|
||||||
|
- flatmap_over_map_reduce
|
||||||
|
- ibinspectable_in_extension
|
||||||
|
- implicit_return
|
||||||
|
- implicitly_unwrapped_optional
|
||||||
|
- joined_default_parameter
|
||||||
|
- last_where
|
||||||
|
- legacy_multiple
|
||||||
|
- let_var_whitespace
|
||||||
|
- literal_expression_end_indentation
|
||||||
|
- lower_acl_than_parent
|
||||||
|
# - missing_docs
|
||||||
|
- modifier_order
|
||||||
|
- multiline_arguments
|
||||||
|
- multiline_arguments_brackets
|
||||||
|
- multiline_function_chains
|
||||||
|
- multiline_literal_brackets
|
||||||
|
- multiline_parameters
|
||||||
|
- multiline_parameters_brackets
|
||||||
|
- nimble_operator
|
||||||
|
- no_extension_access_modifier
|
||||||
|
- no_grouping_extension
|
||||||
|
- nslocalizedstring_key
|
||||||
|
- nslocalizedstring_require_bundle
|
||||||
|
- number_separator
|
||||||
|
- operator_usage_whitespace
|
||||||
|
- optional_enum_case_matching
|
||||||
|
- overridden_super_call
|
||||||
|
- override_in_extension
|
||||||
|
- pattern_matching_keywords
|
||||||
|
- prefer_self_in_static_references
|
||||||
|
- prefer_self_type_over_type_of_self
|
||||||
|
- prefer_zero_over_explicit_init
|
||||||
|
- prefixed_toplevel_constant
|
||||||
|
- private_action
|
||||||
|
- private_outlet
|
||||||
|
- prohibited_interface_builder
|
||||||
|
- prohibited_super_call
|
||||||
|
- quick_discouraged_call
|
||||||
|
- quick_discouraged_focused_test
|
||||||
|
- quick_discouraged_pending_test
|
||||||
|
- redundant_nil_coalescing
|
||||||
|
- redundant_type_annotation
|
||||||
|
- required_enum_case
|
||||||
|
- return_value_from_void_function
|
||||||
|
- self_binding
|
||||||
|
- shorthand_optional_binding
|
||||||
|
- single_test_class
|
||||||
|
- sorted_first_last
|
||||||
|
- sorted_imports
|
||||||
|
- strong_iboutlet
|
||||||
|
- test_case_accessibility
|
||||||
|
- toggle_bool
|
||||||
|
- trailing_closure
|
||||||
|
- unavailable_function
|
||||||
|
- unneeded_parentheses_in_closure_argument
|
||||||
|
- unowned_variable_capture
|
||||||
|
- untyped_error_in_catch
|
||||||
|
- vertical_parameter_alignment_on_call
|
||||||
|
- vertical_whitespace_between_cases
|
||||||
|
- vertical_whitespace_closing_braces
|
||||||
|
- weak_delegate
|
||||||
|
- xct_specific_matcher
|
||||||
|
|
||||||
|
custom_rules:
|
||||||
|
|
||||||
|
# Empty line before and after MARK -------------------------------------------
|
||||||
|
# mark_spacing:
|
||||||
|
# name: "Surround MARK by empty lines"
|
||||||
|
# regex: '\n[^\n]([^\n]*\/\/ MARK[^\n]*)\n[^\n]'
|
||||||
|
# message: "Surround MARK by empty lines"
|
||||||
|
# severity: warning
|
||||||
|
|
||||||
|
# Empty line -----------------------------------------------------------------
|
||||||
|
# no_empty_line_after_func:
|
||||||
|
# name: "No empty line after init or func"
|
||||||
|
# regex: '(func|init|let\s|var\s)[^\n]*\{[^\n\{\}]*\n\n'
|
||||||
|
# message: "No empty line after init or func"
|
||||||
|
# severity: warning
|
||||||
|
|
||||||
|
# Empty line after canImport -----------------------------------------------------------------
|
||||||
|
no_empty_line_after_can_import:
|
||||||
|
name: "Add empty line after #if canImport"
|
||||||
|
regex: '#if canImport\(.*\)\n[^\n]*(import|class|struct|enum|extension|protocol)'
|
||||||
|
message: "Add empty line after #if canImport"
|
||||||
|
severity: warning
|
||||||
|
|
||||||
|
# Spacings -------------------------------------------------------------------
|
||||||
|
empty_line_required:
|
||||||
|
name: "Add empty line after class, struct, enum, extension or protocol"
|
||||||
|
regex: '(class |struct |enum |extension |protocol )[^\n]*\{\n[^\n]*(class|struct|enum|extension|protocol|func|let|var|weak|private|internal|public|open|static|final|\/\/|init|case|@)'
|
||||||
|
message: "Add empty line after class, struct, enum, extension or protocol"
|
||||||
|
severity: warning
|
||||||
|
match_kinds:
|
||||||
|
- argument
|
||||||
|
- attribute.builtin
|
||||||
|
- attribute.id
|
||||||
|
- buildconfig.id
|
||||||
|
- buildconfig.keyword
|
||||||
|
- comment
|
||||||
|
- comment.mark
|
||||||
|
- comment.url
|
||||||
|
- identifier
|
||||||
|
- keyword
|
||||||
|
- number
|
||||||
|
- objectliteral
|
||||||
|
- parameter
|
||||||
|
- placeholder
|
||||||
|
- string
|
||||||
|
- string_interpolation_anchor
|
||||||
|
- typeidentifier
|
||||||
|
|
||||||
|
|
||||||
|
@@ -13,7 +13,7 @@
|
|||||||
- Update plist `UIAppFonts` when generated fonts (use plistBuddy)
|
- Update plist `UIAppFonts` when generated fonts (use plistBuddy)
|
||||||
- New parameter: `infoPlistPaths`
|
- New parameter: `infoPlistPaths`
|
||||||
- Generate SwiftUI extensions for colors, fonts and images
|
- Generate SwiftUI extensions for colors, fonts and images
|
||||||
- New parameter: `extensionNameSwiftUI`
|
- New parameter: `extensionNameUIKit`
|
||||||
- Adding Makefile to install, unsintall and create man page.
|
- Adding Makefile to install, unsintall and create man page.
|
||||||
|
|
||||||
## Fixes
|
## Fixes
|
||||||
|
6
Jenkinsfile
vendored
@@ -1,8 +1,10 @@
|
|||||||
library "openiumpipeline"
|
library "openiumpipeline"
|
||||||
|
|
||||||
env.DEVELOPER_DIR="/Applications/Xcode-15.0.1.app/Contents/Developer"
|
env.DEVELOPER_DIR="/Applications/Xcode-16.3.0.app/Contents/Developer"
|
||||||
//env.SIMULATOR_DEVICE_TYPES="iPad--7th-generation-"
|
// env.SIMULATOR_DEVICE_TYPES="iPhone-14-Pro"
|
||||||
|
// env.SLACK_CHANNEL = "prj-skdevkit"
|
||||||
env.IS_PACKAGE_SWIFT=1
|
env.IS_PACKAGE_SWIFT=1
|
||||||
env.TARGETS_MACOS=1
|
env.TARGETS_MACOS=1
|
||||||
|
env.PACKAGE_NAME="ResgenSwift" // xcodebuild -list => Only 1 scheme
|
||||||
|
|
||||||
iOSpipeline()
|
iOSpipeline()
|
||||||
|
7
Makefile
@@ -1,7 +1,6 @@
|
|||||||
SHELL = /bin/zsh
|
SHELL = /bin/zsh
|
||||||
|
|
||||||
#INSTALL_DIR ?= /usr/local/bin
|
INSTALL_DIR = /usr/local/bin
|
||||||
INSTALL_DIR = /tmp/ResgenYolo
|
|
||||||
|
|
||||||
MAN_DIR := /usr/local/share/man
|
MAN_DIR := /usr/local/share/man
|
||||||
MAN_PAGE_NAME = resgen-swift.1
|
MAN_PAGE_NAME = resgen-swift.1
|
||||||
@@ -14,7 +13,7 @@ BUILD_DIR = $(REPO_DIR)/.build
|
|||||||
|
|
||||||
# create-man-files:
|
# create-man-files:
|
||||||
# swift package plugin generate-manual
|
# swift package plugin generate-manual
|
||||||
# cp $(BUILDDIR)/plugins/GenerateManualPlugin/outputs/ResgenSwift/resgen-swift.1 $(REPODIR)/man/resgen-swift.1
|
# cp $(BUILDDIR)/plugins/GenerateManual/outputs/ResgenSwift/resgen-swift.1 $(REPODIR)/man/resgen-swift.1
|
||||||
|
|
||||||
# install-man-files:
|
# install-man-files:
|
||||||
# mkdir -p ${DESTDIR}${mandir}/man1
|
# mkdir -p ${DESTDIR}${mandir}/man1
|
||||||
@@ -23,7 +22,7 @@ BUILD_DIR = $(REPO_DIR)/.build
|
|||||||
create-and-install-man-files:
|
create-and-install-man-files:
|
||||||
swift package plugin generate-manual
|
swift package plugin generate-manual
|
||||||
mkdir -p ${MAN_DIR}/man1
|
mkdir -p ${MAN_DIR}/man1
|
||||||
cp $(BUILD_DIR)/plugins/GenerateManualPlugin/outputs/ResgenSwift/${MAN_PAGE_NAME} ${MAN_DIR}/man1/${MAN_PAGE_NAME}
|
cp $(BUILD_DIR)/plugins/GenerateManual/outputs/ResgenSwift/${MAN_PAGE_NAME} ${MAN_DIR}/man1/${MAN_PAGE_NAME}
|
||||||
|
|
||||||
#
|
#
|
||||||
# Build and install
|
# Build and install
|
||||||
|
@@ -1,12 +1,22 @@
|
|||||||
{
|
{
|
||||||
|
"originHash" : "fae67d9ce14b6664edc4151ad1a55b74d50d7281499f09eeca50b0376d0f837f",
|
||||||
"pins" : [
|
"pins" : [
|
||||||
{
|
{
|
||||||
"identity" : "swift-argument-parser",
|
"identity" : "swift-argument-parser",
|
||||||
"kind" : "remoteSourceControl",
|
"kind" : "remoteSourceControl",
|
||||||
"location" : "https://github.com/apple/swift-argument-parser",
|
"location" : "https://github.com/apple/swift-argument-parser",
|
||||||
"state" : {
|
"state" : {
|
||||||
"revision" : "9f39744e025c7d377987f30b03770805dcb0bcd1",
|
"revision" : "41982a3656a71c768319979febd796c6fd111d5c",
|
||||||
"version" : "1.1.4"
|
"version" : "1.5.0"
|
||||||
|
}
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"identity" : "swiftlintplugin",
|
||||||
|
"kind" : "remoteSourceControl",
|
||||||
|
"location" : "https://github.com/lukepistrol/SwiftLintPlugin",
|
||||||
|
"state" : {
|
||||||
|
"revision" : "8e785a9801fc7961fef3d41ccb81acb3bd3d9735",
|
||||||
|
"version" : "0.60.0"
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
@@ -14,10 +24,10 @@
|
|||||||
"kind" : "remoteSourceControl",
|
"kind" : "remoteSourceControl",
|
||||||
"location" : "https://github.com/jpsim/Yams.git",
|
"location" : "https://github.com/jpsim/Yams.git",
|
||||||
"state" : {
|
"state" : {
|
||||||
"revision" : "01835dc202670b5bb90d07f3eae41867e9ed29f6",
|
"revision" : "0d9ee7ea8c4ebd4a489ad7a73d5c6cad55d6fed3",
|
||||||
"version" : "5.0.1"
|
"version" : "5.0.6"
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
],
|
],
|
||||||
"version" : 2
|
"version" : 3
|
||||||
}
|
}
|
||||||
|
@@ -1,15 +1,25 @@
|
|||||||
// swift-tools-version:5.6
|
// swift-tools-version:6.1
|
||||||
// The swift-tools-version declares the minimum version of Swift required to build this package.
|
// The swift-tools-version declares the minimum version of Swift required to build this package.
|
||||||
|
|
||||||
import PackageDescription
|
import PackageDescription
|
||||||
|
|
||||||
let package = Package(
|
let package = Package(
|
||||||
name: "ResgenSwift",
|
name: "ResgenSwift",
|
||||||
platforms: [.macOS(.v12)],
|
platforms: [.macOS(.v14), .iOS(.v15)],
|
||||||
dependencies: [
|
dependencies: [
|
||||||
// Dependencies declare other packages that this package depends on.
|
// Dependencies declare other packages that this package depends on.
|
||||||
.package(url: "https://github.com/apple/swift-argument-parser", from: "1.0.0"),
|
.package(
|
||||||
.package(url: "https://github.com/jpsim/Yams.git", from: "5.0.1")
|
url: "https://github.com/apple/swift-argument-parser",
|
||||||
|
from: "1.5.0"
|
||||||
|
),
|
||||||
|
.package(
|
||||||
|
url: "https://github.com/jpsim/Yams.git",
|
||||||
|
from: "5.0.1"
|
||||||
|
),
|
||||||
|
.package(
|
||||||
|
url: "https://github.com/lukepistrol/SwiftLintPlugin",
|
||||||
|
exact: "0.60.0"
|
||||||
|
),
|
||||||
],
|
],
|
||||||
targets: [
|
targets: [
|
||||||
// Targets are the basic building blocks of a package. A target can define a module or a test suite.
|
// Targets are the basic building blocks of a package. A target can define a module or a test suite.
|
||||||
@@ -18,13 +28,27 @@ let package = Package(
|
|||||||
name: "ResgenSwift",
|
name: "ResgenSwift",
|
||||||
dependencies: [
|
dependencies: [
|
||||||
"ToolCore",
|
"ToolCore",
|
||||||
.product(name: "ArgumentParser", package: "swift-argument-parser"),
|
"Yams",
|
||||||
"Yams"
|
.product(
|
||||||
|
name: "ArgumentParser",
|
||||||
|
package: "swift-argument-parser"
|
||||||
|
)
|
||||||
|
],
|
||||||
|
plugins: [
|
||||||
|
// .plugin(name: "SwiftLint", package: "SwiftLintPlugin")
|
||||||
]
|
]
|
||||||
),
|
),
|
||||||
|
|
||||||
// Helper targets
|
// Helper targets
|
||||||
.target(name: "ToolCore"),
|
.target(
|
||||||
|
name: "ToolCore",
|
||||||
|
dependencies: [
|
||||||
|
.product(
|
||||||
|
name: "ArgumentParser",
|
||||||
|
package: "swift-argument-parser"
|
||||||
|
)
|
||||||
|
]
|
||||||
|
),
|
||||||
|
|
||||||
// Test targets
|
// Test targets
|
||||||
.testTarget(
|
.testTarget(
|
||||||
|
559
README.md
@@ -2,64 +2,122 @@
|
|||||||
|
|
||||||
ResgenSwift is a package, fully written in Swift, to help you automatize ressource update and generation.
|
ResgenSwift is a package, fully written in Swift, to help you automatize ressource update and generation.
|
||||||
|
|
||||||
> 🧐 For all commands, see samples files in `SampleFiles`
|
> 🧐 For all commands, see samples files in `SampleFiles` and use `resgen-swift help` and `resgen-swift help <subcommand>` for detailed help.
|
||||||
|
|
||||||
## Fonts
|
## Fonts
|
||||||
|
|
||||||
Font generator generates an extension of `UIFont` and `Font` (or custom classes). It also prints content of `UIAppFonts` from your project `.plist`. If project `.plist` is specified, it will update `UIAppFonts` content of all `.plist`.
|
Font generator generates an extension of `UIFont` and `Font` (or custom classes). It also prints content of `UIAppFonts` from your project `.plist`. If project `.plist` is specified, it will update `UIAppFonts` content of all `.plist`.
|
||||||
|
|
||||||
iOS required to use the **real name** of the font, this name can be different from its filename. To get the **real name**, it uses `fc-scan`. So, be sure that the `$PATH` contains path of `fc-scan`.
|
iOS required to use the **real name** of the font, this name can be different from its filename. To get the **real name**, it uses `fc-scan`. So, be sure that your `$PATH` variable contains path of `fc-scan`.
|
||||||
|
|
||||||
|
|
||||||
|
```sh
|
||||||
|
USAGE:
|
||||||
|
swift run -c release ResgenSwift fonts [-f] <input-file> --extension-output-path <extension-output-path> \
|
||||||
|
[--static-members <static-members>] \
|
||||||
|
[--visibility <visibility>] \
|
||||||
|
[--extension-name <extension-name>] \
|
||||||
|
[--extension-name-ui-kit <extension-name-ui-kit>] \
|
||||||
|
[--extension-suffix <extension-suffix>] \
|
||||||
|
[--info-plist-paths <info-plist-paths>]
|
||||||
|
|
||||||
|
ARGUMENTS:
|
||||||
|
<input-file> Input files where fonts ared defined.
|
||||||
|
|
||||||
|
OPTIONS:
|
||||||
|
-f, -F Should force generation
|
||||||
|
--static-members <static-members>
|
||||||
|
Tell if it will generate static properties or methods (default: false)
|
||||||
|
--visibility <visibility>
|
||||||
|
Visibility of extension and properties. Possibles values: public, private, package, internal. Default is internal (default: internal)
|
||||||
|
--extension-output-path <extension-output-path>
|
||||||
|
Path where to generate the extension.
|
||||||
|
--extension-name <extension-name>
|
||||||
|
Extension name. If not specified, it will generate an Font extension. (default: Font)
|
||||||
|
--extension-name-ui-kit <extension-name-ui-kit>
|
||||||
|
Extension name. If not specified, no extension will be generated.
|
||||||
|
--extension-suffix <extension-suffix>
|
||||||
|
Extension suffix. Ex: MyApp, it will generate {extensionName}+FontsMyApp.swift
|
||||||
|
--info-plist-paths <info-plist-paths>
|
||||||
|
Info.plist paths (array). Will be used to update UIAppFonts content
|
||||||
|
--version Show the version.
|
||||||
|
-h, --help Show help information.
|
||||||
|
```
|
||||||
|
|
||||||
**Example**
|
**Example**
|
||||||
|
|
||||||
```sh
|
```sh
|
||||||
swift run -c release ResgenSwift fonts $FORCE_FLAG "./Fonts/fonts.txt" \
|
swift run -c release ResgenSwift fonts -f "./Fonts/fonts.txt" \
|
||||||
--extension-output-path "./Fonts/Generated" \
|
--extension-output-path "./Fonts/Generated" \
|
||||||
--extension-name "AppFont" \
|
|
||||||
--extension-name-swift-ui "SUIAppFont" \
|
|
||||||
--extension-suffix "GreatApp" \
|
|
||||||
--static-members true \
|
--static-members true \
|
||||||
|
--visibility "public" \
|
||||||
|
--extension-name "AppFont" \
|
||||||
|
--extension-name-ui-kit "UIAppFont" \
|
||||||
|
--extension-suffix "GreatApp" \
|
||||||
--info-plist-paths "./path/one/to/Info.plist ./path/two/to/Info.plist"
|
--info-plist-paths "./path/one/to/Info.plist ./path/two/to/Info.plist"
|
||||||
```
|
```
|
||||||
|
|
||||||
**Parameters**
|
|
||||||
|
|
||||||
1. `-f`: force generation
|
|
||||||
2. Font input folder, it will search for every `.ttf` and `.otf` files specified in `fonts.txt`
|
|
||||||
3. `--extension-output-path`: path where to generate generated extension
|
|
||||||
4. `--extension-name` *(optional)* : name of the class to add UIKit getters
|
|
||||||
5. `--extension-name-swift-ui` *(optional)* : name of the class to add SwiftUI getters
|
|
||||||
6. `--extension-suffix` *(optional)* : additional text which is added to the filename (ex: `AppFont+GreatApp.swift`)
|
|
||||||
7. `--static-members` *(optional)*: generate static properties or not
|
|
||||||
8. `--info-plist-paths` *(optional)*: array of `.plist`, you can specify multiple `Info.plist` for multiple targets
|
|
||||||
|
|
||||||
|
|
||||||
## Colors
|
## Colors
|
||||||
|
|
||||||
Colors generator generates an extension of `UIColor` (or a custom class) along with colorsets in specified xcassets.
|
Colors generator generates colorsets in specified xcassets and an extension of `Color` (or a custom class) associated to those colorsets. If the extension name is not specified, no extension will be generated.
|
||||||
|
|
||||||
|
|
||||||
```sh
|
```sh
|
||||||
swift run -c release ResgenSwift colors $FORCE_FLAG "./Colors/colors.txt" \
|
USAGE:
|
||||||
--style all \
|
swift run -c release ResgenSwift colors [-f] <input-file> \
|
||||||
--xcassets-path "./Colors/colors.xcassets" \
|
--style <style> \
|
||||||
--extension-output-path "./Colors/Generated/" \
|
--xcassets-path <xcassets-path> \
|
||||||
--extension-name "AppColor" \
|
[--static-members <static-members>] \
|
||||||
--extension-name-swift-ui "SUIAppColor" \
|
[--visibility <visibility>] \
|
||||||
--extension-suffix "GreatApp" \
|
[--asset-bundle <asset-bundle>] \
|
||||||
--static-members true
|
[--extension-output-path <extension-output-path>] \
|
||||||
|
[--extension-name <extension-name>] \
|
||||||
|
[--extension-name-ui-kit <extension-name-ui-kit>] \
|
||||||
|
[--extension-suffix <extension-suffix>]
|
||||||
|
|
||||||
|
ARGUMENTS:
|
||||||
|
<input-file> Input files where colors ared defined.
|
||||||
|
|
||||||
|
OPTIONS:
|
||||||
|
-f, -F Should force generation
|
||||||
|
--style <style> Color style to generate: light for light colors only, or all for dark and light colors (values: light, all)
|
||||||
|
--xcassets-path <xcassets-path>
|
||||||
|
Path of xcassets where to generate colors
|
||||||
|
--static-members <static-members>
|
||||||
|
Tell if it will generate static properties or not (default: false)
|
||||||
|
--visibility <visibility>
|
||||||
|
Visibility of extension and properties. Possibles values: public, private, package, internal. Default is internal (default: internal)
|
||||||
|
--asset-bundle <asset-bundle>
|
||||||
|
Bundle where the asset are generated (default: main)
|
||||||
|
--extension-output-path <extension-output-path>
|
||||||
|
Path where to generate the extension.
|
||||||
|
--extension-name <extension-name>
|
||||||
|
SwiftUI extension name. If not specified, no extension will be generated.
|
||||||
|
--extension-name-ui-kit <extension-name-ui-kit>
|
||||||
|
UIKit extension name. If not specified, no extension will be generated.
|
||||||
|
--extension-suffix <extension-suffix>
|
||||||
|
Extension suffix. Ex: MyApp, it will generate {extensionName}+ColorsMyApp.swift
|
||||||
|
--version Show the version.
|
||||||
|
-h, --help Show help information.
|
||||||
```
|
```
|
||||||
|
|
||||||
**Parameters**
|
|
||||||
|
|
||||||
1. `-f`: force generation
|
**Example**
|
||||||
2. Input colors file
|
|
||||||
3. `--style` can be `all` or `light`
|
|
||||||
4. `--extension-output-path`: path where to generate generated extension
|
|
||||||
5. `--extension-name` *(optional)* : name of the class to add UIKit getters
|
|
||||||
6. `--extension-name-swift-ui` *(optional)* : name of the class to add SwiftUI getters
|
|
||||||
7. `--extension-suffix` *(optional)* : additional text which is added to filename (ex: `AppColor+GreatApp.swift`)
|
|
||||||
8. `--static-members` *(optional)*: generate static properties or not
|
|
||||||
|
|
||||||
|
```sh
|
||||||
|
swift run -c release ResgenSwift colors -f "./Colors/colors.txt" \
|
||||||
|
--style all \
|
||||||
|
--xcassets-path "./Colors/colors.xcassets" \
|
||||||
|
--static-members true \
|
||||||
|
--visibility internal \
|
||||||
|
--asset-bundle "main" \
|
||||||
|
--extension-output-path "./Colors/Generated/" \
|
||||||
|
--extension-name "AppColor \
|
||||||
|
--extension-name-ui-kit "UIAppColor" \
|
||||||
|
--extension-suffix "GreatApp"
|
||||||
|
```
|
||||||
|
|
||||||
|
> ⚠️ Option `extensionOutputPath` will require an `extensionName` or `extensionNameUIKit` to generate extension files. Passing an `extensionName` or `extensionNameUIKit` without `extensionOutputPath` will not generate any extension file.
|
||||||
|
|
||||||
## Strings
|
## Strings
|
||||||
|
|
||||||
@@ -68,96 +126,353 @@ Strings command allows to generate `strings` files along with extensions to acce
|
|||||||
### Twine (not recommended)
|
### Twine (not recommended)
|
||||||
|
|
||||||
```sh
|
```sh
|
||||||
swift run -c release ResgenSwift strings twine $FORCE_FLAG "./Twine/strings.txt" \
|
USAGE:
|
||||||
|
swift run -c release ResgenSwift strings twine [-f] <input-file> \
|
||||||
|
--output-path <output-path> \
|
||||||
|
--langs <langs> \
|
||||||
|
--default-lang <default-lang> \
|
||||||
|
--extension-output-path <extension-output-path>
|
||||||
|
|
||||||
|
ARGUMENTS:
|
||||||
|
<input-file> Input files where strings ared defined.
|
||||||
|
|
||||||
|
OPTIONS:
|
||||||
|
-f, -F Should force generation
|
||||||
|
--output-path <output-path>
|
||||||
|
Path where to strings file.
|
||||||
|
--langs <langs> Langs to generate.
|
||||||
|
--default-lang <default-lang>
|
||||||
|
Default langs.
|
||||||
|
--extension-output-path <extension-output-path>
|
||||||
|
Path where to generate the extension.
|
||||||
|
--version Show the version.
|
||||||
|
-h, --help Show help information.
|
||||||
|
```
|
||||||
|
|
||||||
|
**Example**
|
||||||
|
|
||||||
|
```sh
|
||||||
|
swift run -c release ResgenSwift strings twine -f "./Twine/strings.txt" \
|
||||||
--output-path "./Twine/Generated" \
|
--output-path "./Twine/Generated" \
|
||||||
--langs "fr en en-us" \
|
--langs "fr en en-us" \
|
||||||
--default-lang "en" \
|
--default-lang "en" \
|
||||||
--extension-output-path "./Twine/Generated"
|
--extension-output-path "./Twine/Generated"
|
||||||
```
|
```
|
||||||
|
|
||||||
**Parameters**
|
|
||||||
|
|
||||||
1. `-f`: force generation
|
|
||||||
2. Input translations file (must be Twine formatted)
|
|
||||||
3. `--langs`: langs to generate (string with space between each lang)
|
|
||||||
4. `--default-lang`: default lang that will be in `Base.lproj`. It must be in `langs` as well
|
|
||||||
4. `--extension-output-path`: path where to generate generated extension
|
|
||||||
|
|
||||||
### Stringium (recommended)
|
### Stringium (recommended)
|
||||||
|
|
||||||
```sh
|
```sh
|
||||||
swift run -c release ResgenSwift strings stringium $FORCE_FLAG "./Strings/strings.txt" \
|
USAGE:
|
||||||
--output-path "./Strings/Generated" \
|
swift run -c release ResgenSwift strings stringium [-f] <input-file> \
|
||||||
--langs "fr en en-us" \
|
--output-path <output-path> \
|
||||||
--default-lang "en" \
|
--langs <langs> \
|
||||||
--extension-output-path "./Strings/Generated" \
|
--default-lang <default-lang> \
|
||||||
--extension-name "AppString" \
|
[--tags <tags>] \
|
||||||
--extension-suffix "GreatApp" \
|
[--static-members <static-members>] \
|
||||||
--static-members true
|
[--xc-strings <xc-strings>] \
|
||||||
|
[--visibility <visibility>] \
|
||||||
|
[--asset-bundle <asset-bundle>] \
|
||||||
|
[--extension-output-path <extension-output-path>] \
|
||||||
|
[--extension-name <extension-name>] \
|
||||||
|
[--extension-suffix <extension-suffix>]
|
||||||
|
|
||||||
|
ARGUMENTS:
|
||||||
|
<input-file> Input files where strings are defined.
|
||||||
|
|
||||||
|
OPTIONS:
|
||||||
|
-f, -F Should force generation
|
||||||
|
--output-path <output-path>
|
||||||
|
Path where to find the .xcStrings file or the lproj folders.
|
||||||
|
--langs <langs> Langs to generate.
|
||||||
|
--default-lang <default-lang>
|
||||||
|
Default langs.
|
||||||
|
--tags <tags> Tags to generate. (default: ios iosonly iosOnly notranslation)
|
||||||
|
--static-members <static-members>
|
||||||
|
Generate static properties. False by default (default: false)
|
||||||
|
--xc-strings <xc-strings>
|
||||||
|
Tell if it will generate xcStrings file or lproj file. True by default (default: true)
|
||||||
|
--visibility <visibility>
|
||||||
|
Visibility of extension and properties. Possibles values: public, private, package, internal. Default is internal (default: internal)
|
||||||
|
--asset-bundle <asset-bundle>
|
||||||
|
Bundle where the asset are generated (default: main)
|
||||||
|
--extension-output-path <extension-output-path>
|
||||||
|
Path where to generate the extension.
|
||||||
|
--extension-name <extension-name>
|
||||||
|
Extension name. If not specified, no extension will be generated.
|
||||||
|
--extension-suffix <extension-suffix>
|
||||||
|
Extension suffix: {extensionName}+{extensionSuffix}.swift
|
||||||
|
--version Show the version.
|
||||||
|
-h, --help Show help information.
|
||||||
```
|
```
|
||||||
|
|
||||||
**Parameters**
|
|
||||||
|
|
||||||
1. `-f`: force generation
|
**Example**
|
||||||
2. Input translations file (must be Twine formatted)
|
|
||||||
3. `--langs`: langs to generate (string with space between each lang)
|
|
||||||
4. `--default-lang`: default lang that will be in `Base.lproj`. It must be in `langs` as well
|
|
||||||
4. `--extension-output-path`: path where to generate generated extension
|
|
||||||
5. `--extension-name` *(optional)* : name of class to add the extension
|
|
||||||
6. `--extension-suffix` *(optional)* : additional text which is added to filename (ex: `AppString+GreatApp.swift`)
|
|
||||||
7. `--static-members` *(optional)*: generate static properties or not
|
|
||||||
|
|
||||||
|
```sh
|
||||||
|
swift run -c release ResgenSwift strings stringium -f "./Strings/strings.txt" \
|
||||||
|
--output-path "./Strings/Generated" \
|
||||||
|
--langs "fr en en-us" \
|
||||||
|
--default-lang "en \
|
||||||
|
--tags "ios iosonly iosOnly notranslation" \
|
||||||
|
--static-members true \
|
||||||
|
--xc-strings true \
|
||||||
|
--visibility "package" \
|
||||||
|
--asset-bundle "module" \
|
||||||
|
--extension-output-path "./Strings/Generated" \
|
||||||
|
--extension-name "AppString \
|
||||||
|
--extension-suffix "GreatApp"
|
||||||
|
```
|
||||||
|
|
||||||
|
> ⚠️ Option `extensionOutputPath` will require an `extensionName` or `extensionNameUIKit` to generate extension files. Passing an `extensionName` or `extensionNameUIKit` without `extensionOutputPath` will not generate any extension file.
|
||||||
|
|
||||||
## Tags
|
## Tags
|
||||||
|
|
||||||
Tags is also a subcommand of `Strings`. Input files are formatted the same way. Tags will generate properties which return exactly what is specified in the input file. It was designed to be used for analytics purpose and to be shared with any other platform to have the same analytics keys.
|
Tags is also a subcommand of `Strings`. Input files are formatted the same way. Tags will generate properties which return exactly what is specified in the input file. It was designed to be used for analytics purpose and to be shared with any other platform to have the same analytics keys.
|
||||||
|
|
||||||
```sh
|
```sh
|
||||||
swift run -c release ResgenSwift strings tags $FORCE_FLAG "./Tags/tags.txt" \
|
USAGE:
|
||||||
--lang "ium" \
|
swift run -c release ResgenSwift strings tags [-f] <input-file> \
|
||||||
--extension-output-path "./Tags/Generated" \
|
--extension-output-path <extension-output-path> \
|
||||||
--extension-name "AppTags" \
|
[--lang <lang>] \
|
||||||
--extension-suffix "GreatApp" \
|
[--visibility <visibility>] \
|
||||||
--static-members true
|
[--static-members <static-members>] \
|
||||||
|
[--extension-name <extension-name>] \
|
||||||
|
[--extension-suffix <extension-suffix>]
|
||||||
|
|
||||||
|
ARGUMENTS:
|
||||||
|
<input-file> Input files where tags ared defined.
|
||||||
|
|
||||||
|
OPTIONS:
|
||||||
|
-f, -F Should force generation
|
||||||
|
--lang <lang> Lang to generate. ("ium" by default) (default: ium)
|
||||||
|
--visibility <visibility>
|
||||||
|
Visibility of extension and properties. Possibles values: public, private, package, internal. Default is internal (default: internal)
|
||||||
|
--extension-output-path <extension-output-path>
|
||||||
|
Path where to generate the extension.
|
||||||
|
--static-members <static-members>
|
||||||
|
Tell if it will generate static properties or not (default: false)
|
||||||
|
--extension-name <extension-name>
|
||||||
|
Extension name. If not specified, it will generate a Tag extension. (default: Tags)
|
||||||
|
--extension-suffix <extension-suffix>
|
||||||
|
Extension suffix. Ex: MyApp, it will generate {extensionName}+Tag{extensionSuffix}.swift
|
||||||
|
--version Show the version.
|
||||||
|
-h, --help Show help information.
|
||||||
```
|
```
|
||||||
|
|
||||||
|
**Example**
|
||||||
|
|
||||||
|
```sh
|
||||||
|
swift run -c release ResgenSwift strings tags -f "./Tags/tags.txt" \
|
||||||
|
--extension-output-path "./Tags/Generated" \
|
||||||
|
--lang "ium" \
|
||||||
|
--visibility "public" \
|
||||||
|
--static-members true \
|
||||||
|
--extension-name "AppTags" \
|
||||||
|
--extension-suffix "GreatApp"
|
||||||
|
```
|
||||||
|
|
||||||
|
> ⚠️ If extension name is not set or is `Tags`, it will generate the following typealias `typealias Tags = String`.
|
||||||
|
|
||||||
|
## Analytics
|
||||||
|
|
||||||
|
Analytics will generate all you need to analyze UX with Matomo or Firebase Analytics. Input files are formatted in YAML. This command will generate a manager for each target and an AnalyticsManager. This is this one you will need to use. And it will generate a method for all tags you have declared in the YAML file. Next, you will need to use the `configure()` method of AnalyticsManager and if you want to use matomo to set up the `siteId` and the `url` of the site.
|
||||||
|
|
||||||
|
```sh
|
||||||
|
USAGE:
|
||||||
|
swift run -c release ResgenSwift analytics [-f] <input-file> \
|
||||||
|
--target <target> \
|
||||||
|
--output-file <output-file> \
|
||||||
|
[--static-members <static-members>] \
|
||||||
|
[--visibility <visibility>]
|
||||||
|
|
||||||
|
ARGUMENTS:
|
||||||
|
<input-file> Input files where tags ared defined.
|
||||||
|
|
||||||
|
OPTIONS:
|
||||||
|
-f, -F Should force generation
|
||||||
|
--target <target> Target(s) analytics to generate. ("matomo" | "firebase")
|
||||||
|
--output-file <output-file>
|
||||||
|
Where to generate the analytics manager (path with filename)
|
||||||
|
--static-members <static-members>
|
||||||
|
Tell if it will generate static properties or not (default: false)
|
||||||
|
--visibility <visibility>
|
||||||
|
Visibility of extension and properties. Possibles values: public, private, package, internal. Default is internal (default: internal)
|
||||||
|
--version Show the version.
|
||||||
|
-h, --help Show help information.
|
||||||
|
```
|
||||||
|
|
||||||
|
**Example**
|
||||||
|
|
||||||
|
```sh
|
||||||
|
swift run -c release ResgenSwift analytics -f "./Tags/analytics.yml" \
|
||||||
|
--target "matomo firebase" \
|
||||||
|
--output-file "./Analytics/Generated/AppAnalytics+GreatApp.swift" \
|
||||||
|
--static-members false \
|
||||||
|
--visibility "public"
|
||||||
|
```
|
||||||
|
|
||||||
|
### YAML
|
||||||
|
|
||||||
|
```
|
||||||
|
- id: s1_def_one
|
||||||
|
name: s1 def one _TITLE_
|
||||||
|
path: s1_def_one/_TITLE_
|
||||||
|
action: Tap
|
||||||
|
category: User
|
||||||
|
tags: ios,droid
|
||||||
|
comments:
|
||||||
|
parameters:
|
||||||
|
- name: title
|
||||||
|
type: String
|
||||||
|
replaceIn: name,path
|
||||||
|
```
|
||||||
|
|
||||||
|
1. `id`: name of the method (method name will be composed of `log` + `Event|Screen` + `id`)
|
||||||
|
2. `name`: name of the tag
|
||||||
|
3. `path` *(optional with firebase)* : needed for matomo but not with firebase (log screen)
|
||||||
|
4. `action` *(optional with firebase)* : needed for matomo but not with firebase (log event)
|
||||||
|
5. `category` *(optional with firebase)* : needed for matomo but not with firebase (log event)
|
||||||
|
6. `tags`: which platform target
|
||||||
|
7. `comments` *(optional)*
|
||||||
|
8. `parameters` *(optional)*
|
||||||
|
|
||||||
**Parameters**
|
**Parameters**
|
||||||
|
|
||||||
1. `-f`: force generation
|
You can use parameters in generate methods.
|
||||||
2. Input tags file (must be Twine formatted)
|
|
||||||
3. `--lang`: langs to look at in input file
|
|
||||||
4. `--extension-output-path`: path where to generate generated extension
|
|
||||||
5. `--extension-name` *(optional)* : name of class to add the extension
|
|
||||||
6. `--extension-suffix` *(optional)* : additional text which is added to filename (ex: `AppTags+GreatApp.swift`)
|
|
||||||
7. `--static-members` *(optional)*: generate static properties or not
|
|
||||||
|
|
||||||
> ⚠️ If extension name is not set or is `Tags`, it will generate the following typaloas `typealias Tags = String`.
|
1. `name`: name of the parameter
|
||||||
|
2. `type`: type of the parameter (Int, String, Bool, Double)
|
||||||
|
3. `value`: value of the parameter
|
||||||
|
4. `defaultValue`: defaultValue of the parameter
|
||||||
|
3. `replaceIn` *(optional)*
|
||||||
|
|
||||||
|
**Value**
|
||||||
|
|
||||||
|
If you want to send another parameter with a static value. For example, you want to send to which screen the event is triggered.
|
||||||
|
You can add the parameter 'screenName' for example and its 'value' is 'Home'. With this, you do not need to specify the value in the function call.
|
||||||
|
|
||||||
|
**DefaultValue**
|
||||||
|
|
||||||
|
If you want ta add a parameter in the call of the function but you want to make it optionnal with a default value you need to use this property.
|
||||||
|
|
||||||
|
Example:
|
||||||
|
```
|
||||||
|
events:
|
||||||
|
id: id_of_tag
|
||||||
|
name: _TITLE_
|
||||||
|
tags: ios,droid
|
||||||
|
parameters:
|
||||||
|
- name: title
|
||||||
|
type: String
|
||||||
|
defaultValue: someTitle
|
||||||
|
```
|
||||||
|
|
||||||
|
The generated method will be:
|
||||||
|
```
|
||||||
|
logIdOfTag(title: String = "someTitle")
|
||||||
|
```
|
||||||
|
|
||||||
|
**Replace in**
|
||||||
|
|
||||||
|
This is section is equivalent of `%s | %d | %f | %@`. You can put the content of the parameter in *name*, *path*, *action*, *category*.
|
||||||
|
You need to put `_` + `NAME OF THE PARAMETER` + `_` in the target and which target you want in the value of `replaceIn`. (name need to be in uppercase).
|
||||||
|
You can't use `value`and `replaceIn`in thge same time.
|
||||||
|
|
||||||
|
Example:
|
||||||
|
```
|
||||||
|
events:
|
||||||
|
id: id_of_tag
|
||||||
|
name: _TITLE_
|
||||||
|
tags: ios,droid
|
||||||
|
parameters:
|
||||||
|
- name: title
|
||||||
|
type: String
|
||||||
|
replaceIn: name
|
||||||
|
```
|
||||||
|
|
||||||
|
In this sample, we want to add the parameter `title` in the field `name`. So, we need to place `_TITLE_` in the field `name`.
|
||||||
|
|
||||||
|
The generated method will be:
|
||||||
|
```
|
||||||
|
logIdOfTag(title: String)
|
||||||
|
```
|
||||||
|
|
||||||
|
You can also want to replace a parameter in an other parameter. You can do this with the `replaceIn` property. The condition is that the parameter which will use the `replaceIn`need to have the `value`property
|
||||||
|
|
||||||
|
Example:
|
||||||
|
```
|
||||||
|
events:
|
||||||
|
id: id_of_tag
|
||||||
|
name: title
|
||||||
|
tags: ios,droid
|
||||||
|
parameters:
|
||||||
|
- name: something
|
||||||
|
type: String
|
||||||
|
value: test _TEXT_
|
||||||
|
- name: text
|
||||||
|
type: String
|
||||||
|
replaceIn: something
|
||||||
|
```
|
||||||
|
|
||||||
## Images
|
## Images
|
||||||
|
|
||||||
Images generator will generate images assets along with extensions to access those images easily.
|
Images generator will generate images assets along with extensions to access those images easily. If the extension name is not specified, no extension will be generated.
|
||||||
|
|
||||||
```sh
|
```sh
|
||||||
swift run -c release ResgenSwift images $FORCE_FLAG "./Images/images.txt" \
|
USAGE:
|
||||||
--xcassets-path "./Images/app.xcassets" \
|
resgen-swift images [-f] [-F] <input-file> \
|
||||||
--extension-output-path "./Images/Generated" \
|
--xcassets-path <xcassets-path> \
|
||||||
--extension-name "AppImage" \
|
[--static-members <static-members>] \
|
||||||
--extension-name-swift-ui "SUIAppImage" \
|
[--visibility <visibility>] \
|
||||||
--extension-suffix "GreatApp" \
|
[--asset-bundle <asset-bundle>] \
|
||||||
--static-members true
|
[--extension-output-path <extension-output-path>] \
|
||||||
|
[--extension-name <extension-name>] \
|
||||||
|
[--extension-name-ui-kit <extension-name-ui-kit>] \
|
||||||
|
[--extension-suffix <extension-suffix>]
|
||||||
|
|
||||||
|
ARGUMENTS:
|
||||||
|
<input-file> Input files where strings ared defined.
|
||||||
|
|
||||||
|
OPTIONS:
|
||||||
|
-f Should force script execution
|
||||||
|
-F Regenerate all images
|
||||||
|
--xcassets-path <xcassets-path>
|
||||||
|
Xcassets path where to generate images.
|
||||||
|
--static-members <static-members>
|
||||||
|
Tell if it will generate static properties or not (default: false)
|
||||||
|
--visibility <visibility>
|
||||||
|
Visibility of extension and properties. Possibles values: public, private, package, internal. Default is internal (default: internal)
|
||||||
|
--asset-bundle <asset-bundle>
|
||||||
|
Bundle where the asset are generated (default: main)
|
||||||
|
--extension-output-path <extension-output-path>
|
||||||
|
Path where to generate the extension.
|
||||||
|
--extension-name <extension-name>
|
||||||
|
SwiftUI extension name. If not specified, no extension will be generated.
|
||||||
|
--extension-name-ui-kit <extension-name-ui-kit>
|
||||||
|
UIKit extension name. If not specified, no extension will be generated.
|
||||||
|
--extension-suffix <extension-suffix>
|
||||||
|
Extension suffix. Ex: MyApp, it will generate {extensionName}+Image{extensionSuffix}.swift
|
||||||
|
--version Show the version.
|
||||||
|
-h, --help Show help information.
|
||||||
```
|
```
|
||||||
|
|
||||||
**Parameters**
|
**Example**
|
||||||
|
|
||||||
1. `-f`: force generation
|
```sh
|
||||||
2. Input images definitions file
|
swift run -c release ResgenSwift images -F "./Images/images.txt" \
|
||||||
3. `--xcassets-path`: xcasset path where to generate imagesets
|
--xcassets-path "./Images/app.xcassets" \
|
||||||
4. `--extension-output-path`: path where to generate generated extension
|
--static-members false \
|
||||||
5. `--extension-name` *(optional)* : name of the class to add UIKit getters
|
--visibility "public" \
|
||||||
6. `--extension-name-swift-ui` *(optional)* : name of the class to add SwiftUI getters
|
--asset-bundle "module" \
|
||||||
6. `--extension-suffix` *(optional)* : additional text which is added to filename (ex: `AppImage+GreatApp.swift`)
|
--extension-output-path "./Images/Generated" \
|
||||||
7. `--static-members` *(optional)*: generate static properties or not
|
--extension-name "AppImage" \
|
||||||
|
--extension-name-ui-kit "UIAppImage" \
|
||||||
|
--extension-suffix "GreatApp"
|
||||||
|
```
|
||||||
|
|
||||||
|
> ⚠️ Option `extensionOutputPath` will require an `extensionName` or `extensionNameUIKit` to generate extension files. Passing an `extensionName` or `extensionNameUIKit` without `extensionOutputPath` will not generate any extension file.
|
||||||
|
|
||||||
|
> ⚠️ Svg images will be copied in the assets and rendered as "Original", however if those images are not rendered correctly you can force the png generation by adding the key word "png" like this: id arrow_back 15 ? png
|
||||||
|
|
||||||
## All at once
|
## All at once
|
||||||
|
|
||||||
@@ -172,22 +487,25 @@ All parameters can be specified in a configuration file in `Yaml`:
|
|||||||
colors:
|
colors:
|
||||||
-
|
-
|
||||||
inputFile: String
|
inputFile: String
|
||||||
style: [light/all]
|
style: String
|
||||||
xcassetsPath: String
|
xcassetsPath: String
|
||||||
extensionOutputPath: String
|
extensionOutputPath: String
|
||||||
extensionName: String?
|
extensionName: String?
|
||||||
extensionNameSwiftUI: String?
|
extensionNameUIKit: String?
|
||||||
extensionSuffix: String?
|
extensionSuffix: String?
|
||||||
staticMembers: Bool?
|
staticMembers: Bool?
|
||||||
|
visiblity: String?
|
||||||
|
assetBundle: String?
|
||||||
|
|
||||||
fonts:
|
fonts:
|
||||||
-
|
-
|
||||||
inputFile: String
|
inputFile: String
|
||||||
extensionOutputPath: String
|
extensionOutputPath: String
|
||||||
extensionName: String?
|
extensionName: String?
|
||||||
extensionNameSwiftUI: String?
|
extensionNameUIKit: String?
|
||||||
extensionSuffix: String?
|
extensionSuffix: String?
|
||||||
staticMembers: Bool?
|
staticMembers: Bool?
|
||||||
|
visiblity: String?
|
||||||
infoPlistPaths: [String]
|
infoPlistPaths: [String]
|
||||||
|
|
||||||
images:
|
images:
|
||||||
@@ -196,9 +514,11 @@ images:
|
|||||||
xcassetsPath: String
|
xcassetsPath: String
|
||||||
extensionOutputPath: String
|
extensionOutputPath: String
|
||||||
extensionName: String?
|
extensionName: String?
|
||||||
extensionNameSwiftUI: String?
|
extensionNameUIKit: String?
|
||||||
extensionSuffix: String?
|
extensionSuffix: String?
|
||||||
staticMembers: Bool?
|
staticMembers: Bool?
|
||||||
|
visiblity: String?
|
||||||
|
assetBundle: String?
|
||||||
|
|
||||||
strings:
|
strings:
|
||||||
-
|
-
|
||||||
@@ -210,6 +530,9 @@ strings:
|
|||||||
extensionName: String?
|
extensionName: String?
|
||||||
extensionSuffix: String?
|
extensionSuffix: String?
|
||||||
staticMembers: Bool?
|
staticMembers: Bool?
|
||||||
|
visiblity: String?
|
||||||
|
assetBundle: String?
|
||||||
|
xcStrings: Bool?
|
||||||
|
|
||||||
tags:
|
tags:
|
||||||
-
|
-
|
||||||
@@ -219,6 +542,15 @@ tags:
|
|||||||
extensionName: String?
|
extensionName: String?
|
||||||
extensionSuffix: String?
|
extensionSuffix: String?
|
||||||
staticMembers: Bool?
|
staticMembers: Bool?
|
||||||
|
visiblity: String?
|
||||||
|
|
||||||
|
analytics:
|
||||||
|
-
|
||||||
|
inputFile: String
|
||||||
|
target: String
|
||||||
|
outputFile: String
|
||||||
|
visibility: String?
|
||||||
|
staticMembers: Bool?
|
||||||
```
|
```
|
||||||
|
|
||||||
### Multiple configurations
|
### Multiple configurations
|
||||||
@@ -232,22 +564,26 @@ Sample for 2 colors configurations:
|
|||||||
colors:
|
colors:
|
||||||
-
|
-
|
||||||
inputFile: String
|
inputFile: String
|
||||||
style: [light/all]
|
style: String
|
||||||
xcassetsPath: String
|
xcassetsPath: String
|
||||||
extensionOutputPath: String
|
extensionOutputPath: String
|
||||||
extensionName: String?
|
extensionName: String?
|
||||||
extensionNameSwiftUI: String?
|
extensionNameUIKit: String?
|
||||||
extensionSuffix: String?
|
extensionSuffix: String?
|
||||||
staticMembers: Bool?
|
staticMembers: Bool?
|
||||||
|
visiblity: String?
|
||||||
|
assetBundle: String?
|
||||||
-
|
-
|
||||||
inputFile: String
|
inputFile: String
|
||||||
style: [light/all]
|
style: String
|
||||||
xcassetsPath: String
|
xcassetsPath: String
|
||||||
extensionOutputPath: String
|
extensionOutputPath: String
|
||||||
extensionName: String?
|
extensionName: String?
|
||||||
extensionNameSwiftUI: String?
|
extensionNameUIKit: String?
|
||||||
extensionSuffix: String?
|
extensionSuffix: String?
|
||||||
staticMembers: Bool?
|
staticMembers: Bool?
|
||||||
|
visiblity: String?
|
||||||
|
assetBundle: String?
|
||||||
...
|
...
|
||||||
```
|
```
|
||||||
|
|
||||||
@@ -291,8 +627,9 @@ architecture:
|
|||||||
```
|
```
|
||||||
|
|
||||||
This will generate a file named as the architecture classname: `R.swift`. Based on the previous architecture, it will generate:
|
This will generate a file named as the architecture classname: `R.swift`. Based on the previous architecture, it will generate:
|
||||||
```
|
|
||||||
class R {
|
```swift
|
||||||
|
class R: Sendable {
|
||||||
static let images = R2Image()
|
static let images = R2Image()
|
||||||
static let strings = R2String()
|
static let strings = R2String()
|
||||||
static let fonts = R2Font()
|
static let fonts = R2Font()
|
||||||
@@ -300,25 +637,25 @@ class R {
|
|||||||
static let uikit = R2UI()
|
static let uikit = R2UI()
|
||||||
}
|
}
|
||||||
|
|
||||||
class R2Image {}
|
class R2Image: Sendable {}
|
||||||
|
|
||||||
class R2String {}
|
class R2String: Sendable {}
|
||||||
|
|
||||||
class R2Font {}
|
class R2Font: Sendable {}
|
||||||
|
|
||||||
class R2Image {}
|
class R2Image: Sendable {}
|
||||||
|
|
||||||
class R2UI {
|
class R2UI: Sendable {
|
||||||
let images = R2UIImage()
|
let images = R2UIImage()
|
||||||
let fonts = R2UIFont()
|
let fonts = R2UIFont()
|
||||||
let images = R2UIImage()
|
let images = R2UIImage()
|
||||||
}
|
}
|
||||||
|
|
||||||
class R2UIImage {}
|
class R2UIImage: Sendable {}
|
||||||
|
|
||||||
class R2UIFont {}
|
class R2UIFont: Sendable {}
|
||||||
|
|
||||||
class R2UIImage {}
|
class R2UIImage: Sendable {}
|
||||||
```
|
```
|
||||||
|
|
||||||
|
|
||||||
|
@@ -1,21 +1,21 @@
|
|||||||
// Generated by ResgenSwift.Color 1.2
|
// Generated by ResgenSwift.Color 2.2.0
|
||||||
|
|
||||||
import SwiftUI
|
import SwiftUI
|
||||||
|
|
||||||
extension ColorYolo {
|
extension ColorYolo {
|
||||||
|
|
||||||
/// Color red is #FF0000 (light) or #FF0000 (dark)"
|
/// Color red is #FF0000 (light) or #FF0000 (dark)"
|
||||||
var red: Color {
|
public var red: Color {
|
||||||
Color("red")
|
Color("red")
|
||||||
}
|
}
|
||||||
|
|
||||||
/// Color green_alpha_50 is #A000FF00 (light) or #A000FF00 (dark)"
|
/// Color green_alpha_50 is #A000FF00 (light) or #A000FF00 (dark)"
|
||||||
var green_alpha_50: Color {
|
public var green_alpha_50: Color {
|
||||||
Color("green_alpha_50")
|
Color("green_alpha_50")
|
||||||
}
|
}
|
||||||
|
|
||||||
/// Color blue_light_dark is #0000FF (light) or #0000AA (dark)"
|
/// Color blue_light_dark is #0000FF (light) or #0000AA (dark)"
|
||||||
var blue_light_dark: Color {
|
public var blue_light_dark: Color {
|
||||||
Color("blue_light_dark")
|
Color("blue_light_dark")
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@@ -1,21 +1,21 @@
|
|||||||
// Generated by ResgenSwift.Color 1.2
|
// Generated by ResgenSwift.Color 2.2.0
|
||||||
|
|
||||||
import UIKit
|
import UIKit
|
||||||
|
|
||||||
extension UIColorYolo {
|
extension UIColorYolo {
|
||||||
|
|
||||||
/// Color red is #FF0000 (light) or #FF0000 (dark)"
|
/// Color red is #FF0000 (light) or #FF0000 (dark)"
|
||||||
@objc var red: UIColor {
|
@objc public var red: UIColor {
|
||||||
UIColor(named: "red")!
|
UIColor(named: "red")!
|
||||||
}
|
}
|
||||||
|
|
||||||
/// Color green_alpha_50 is #A000FF00 (light) or #A000FF00 (dark)"
|
/// Color green_alpha_50 is #A000FF00 (light) or #A000FF00 (dark)"
|
||||||
@objc var green_alpha_50: UIColor {
|
@objc public var green_alpha_50: UIColor {
|
||||||
UIColor(named: "green_alpha_50")!
|
UIColor(named: "green_alpha_50")!
|
||||||
}
|
}
|
||||||
|
|
||||||
/// Color blue_light_dark is #0000FF (light) or #0000AA (dark)"
|
/// Color blue_light_dark is #0000FF (light) or #0000AA (dark)"
|
||||||
@objc var blue_light_dark: UIColor {
|
@objc public var blue_light_dark: UIColor {
|
||||||
UIColor(named: "blue_light_dark")!
|
UIColor(named: "blue_light_dark")!
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@@ -0,0 +1,21 @@
|
|||||||
|
// Generated by ResgenSwift.Color 2.1.0
|
||||||
|
|
||||||
|
import UIKit
|
||||||
|
|
||||||
|
extension UIhkjhkColorYolo {
|
||||||
|
|
||||||
|
/// Color red is #FF0000 (light) or #FF0000 (dark)"
|
||||||
|
@objc var red: UIColor {
|
||||||
|
UIColor(named: "red")!
|
||||||
|
}
|
||||||
|
|
||||||
|
/// Color green_alpha_50 is #A000FF00 (light) or #A000FF00 (dark)"
|
||||||
|
@objc var green_alpha_50: UIColor {
|
||||||
|
UIColor(named: "green_alpha_50")!
|
||||||
|
}
|
||||||
|
|
||||||
|
/// Color blue_light_dark is #0000FF (light) or #0000AA (dark)"
|
||||||
|
@objc var blue_light_dark: UIColor {
|
||||||
|
UIColor(named: "blue_light_dark")!
|
||||||
|
}
|
||||||
|
}
|
@@ -1,10 +1,10 @@
|
|||||||
// Generated by ResgenSwift.Fonts 1.2
|
// Generated by ResgenSwift.Fonts 2.2.0
|
||||||
|
|
||||||
import SwiftUI
|
import SwiftUI
|
||||||
|
|
||||||
extension FontYolo {
|
extension FontYolo {
|
||||||
|
|
||||||
enum FontName: String {
|
public enum FontName: String {
|
||||||
case LatoItalic = "Lato-Italic"
|
case LatoItalic = "Lato-Italic"
|
||||||
case LatoLightItalic = "Lato-LightItalic"
|
case LatoLightItalic = "Lato-LightItalic"
|
||||||
case LatoHairline = "Lato-Hairline"
|
case LatoHairline = "Lato-Hairline"
|
||||||
@@ -19,43 +19,43 @@ extension FontYolo {
|
|||||||
|
|
||||||
// MARK: - Getter
|
// MARK: - Getter
|
||||||
|
|
||||||
func LatoItalic(withSize size: CGFloat) -> Font {
|
public func LatoItalic(withSize size: CGFloat) -> Font {
|
||||||
Font.custom(FontName.LatoItalic.rawValue, size: size)
|
Font.custom(FontName.LatoItalic.rawValue, size: size)
|
||||||
}
|
}
|
||||||
|
|
||||||
func LatoLightItalic(withSize size: CGFloat) -> Font {
|
public func LatoLightItalic(withSize size: CGFloat) -> Font {
|
||||||
Font.custom(FontName.LatoLightItalic.rawValue, size: size)
|
Font.custom(FontName.LatoLightItalic.rawValue, size: size)
|
||||||
}
|
}
|
||||||
|
|
||||||
func LatoHairline(withSize size: CGFloat) -> Font {
|
public func LatoHairline(withSize size: CGFloat) -> Font {
|
||||||
Font.custom(FontName.LatoHairline.rawValue, size: size)
|
Font.custom(FontName.LatoHairline.rawValue, size: size)
|
||||||
}
|
}
|
||||||
|
|
||||||
func LatoBold(withSize size: CGFloat) -> Font {
|
public func LatoBold(withSize size: CGFloat) -> Font {
|
||||||
Font.custom(FontName.LatoBold.rawValue, size: size)
|
Font.custom(FontName.LatoBold.rawValue, size: size)
|
||||||
}
|
}
|
||||||
|
|
||||||
func LatoBlack(withSize size: CGFloat) -> Font {
|
public func LatoBlack(withSize size: CGFloat) -> Font {
|
||||||
Font.custom(FontName.LatoBlack.rawValue, size: size)
|
Font.custom(FontName.LatoBlack.rawValue, size: size)
|
||||||
}
|
}
|
||||||
|
|
||||||
func LatoRegular(withSize size: CGFloat) -> Font {
|
public func LatoRegular(withSize size: CGFloat) -> Font {
|
||||||
Font.custom(FontName.LatoRegular.rawValue, size: size)
|
Font.custom(FontName.LatoRegular.rawValue, size: size)
|
||||||
}
|
}
|
||||||
|
|
||||||
func LatoBlackItalic(withSize size: CGFloat) -> Font {
|
public func LatoBlackItalic(withSize size: CGFloat) -> Font {
|
||||||
Font.custom(FontName.LatoBlackItalic.rawValue, size: size)
|
Font.custom(FontName.LatoBlackItalic.rawValue, size: size)
|
||||||
}
|
}
|
||||||
|
|
||||||
func LatoBoldItalic(withSize size: CGFloat) -> Font {
|
public func LatoBoldItalic(withSize size: CGFloat) -> Font {
|
||||||
Font.custom(FontName.LatoBoldItalic.rawValue, size: size)
|
Font.custom(FontName.LatoBoldItalic.rawValue, size: size)
|
||||||
}
|
}
|
||||||
|
|
||||||
func LatoLight(withSize size: CGFloat) -> Font {
|
public func LatoLight(withSize size: CGFloat) -> Font {
|
||||||
Font.custom(FontName.LatoLight.rawValue, size: size)
|
Font.custom(FontName.LatoLight.rawValue, size: size)
|
||||||
}
|
}
|
||||||
|
|
||||||
func LatoHairlineItalic(withSize size: CGFloat) -> Font {
|
public func LatoHairlineItalic(withSize size: CGFloat) -> Font {
|
||||||
Font.custom(FontName.LatoHairlineItalic.rawValue, size: size)
|
Font.custom(FontName.LatoHairlineItalic.rawValue, size: size)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@@ -1,10 +1,10 @@
|
|||||||
// Generated by ResgenSwift.Fonts 1.2
|
// Generated by ResgenSwift.Fonts 2.2.0
|
||||||
|
|
||||||
import UIKit
|
import UIKit
|
||||||
|
|
||||||
extension UIFontYolo {
|
extension UIFontYolo {
|
||||||
|
|
||||||
enum FontName: String {
|
public enum FontName: String {
|
||||||
case LatoItalic = "Lato-Italic"
|
case LatoItalic = "Lato-Italic"
|
||||||
case LatoLightItalic = "Lato-LightItalic"
|
case LatoLightItalic = "Lato-LightItalic"
|
||||||
case LatoHairline = "Lato-Hairline"
|
case LatoHairline = "Lato-Hairline"
|
||||||
@@ -19,43 +19,43 @@ extension UIFontYolo {
|
|||||||
|
|
||||||
// MARK: - Getter
|
// MARK: - Getter
|
||||||
|
|
||||||
func LatoItalic(withSize size: CGFloat) -> UIFont {
|
public func LatoItalic(withSize size: CGFloat) -> UIFont {
|
||||||
UIFont(name: FontName.LatoItalic.rawValue, size: size)!
|
UIFont(name: FontName.LatoItalic.rawValue, size: size)!
|
||||||
}
|
}
|
||||||
|
|
||||||
func LatoLightItalic(withSize size: CGFloat) -> UIFont {
|
public func LatoLightItalic(withSize size: CGFloat) -> UIFont {
|
||||||
UIFont(name: FontName.LatoLightItalic.rawValue, size: size)!
|
UIFont(name: FontName.LatoLightItalic.rawValue, size: size)!
|
||||||
}
|
}
|
||||||
|
|
||||||
func LatoHairline(withSize size: CGFloat) -> UIFont {
|
public func LatoHairline(withSize size: CGFloat) -> UIFont {
|
||||||
UIFont(name: FontName.LatoHairline.rawValue, size: size)!
|
UIFont(name: FontName.LatoHairline.rawValue, size: size)!
|
||||||
}
|
}
|
||||||
|
|
||||||
func LatoBold(withSize size: CGFloat) -> UIFont {
|
public func LatoBold(withSize size: CGFloat) -> UIFont {
|
||||||
UIFont(name: FontName.LatoBold.rawValue, size: size)!
|
UIFont(name: FontName.LatoBold.rawValue, size: size)!
|
||||||
}
|
}
|
||||||
|
|
||||||
func LatoBlack(withSize size: CGFloat) -> UIFont {
|
public func LatoBlack(withSize size: CGFloat) -> UIFont {
|
||||||
UIFont(name: FontName.LatoBlack.rawValue, size: size)!
|
UIFont(name: FontName.LatoBlack.rawValue, size: size)!
|
||||||
}
|
}
|
||||||
|
|
||||||
func LatoRegular(withSize size: CGFloat) -> UIFont {
|
public func LatoRegular(withSize size: CGFloat) -> UIFont {
|
||||||
UIFont(name: FontName.LatoRegular.rawValue, size: size)!
|
UIFont(name: FontName.LatoRegular.rawValue, size: size)!
|
||||||
}
|
}
|
||||||
|
|
||||||
func LatoBlackItalic(withSize size: CGFloat) -> UIFont {
|
public func LatoBlackItalic(withSize size: CGFloat) -> UIFont {
|
||||||
UIFont(name: FontName.LatoBlackItalic.rawValue, size: size)!
|
UIFont(name: FontName.LatoBlackItalic.rawValue, size: size)!
|
||||||
}
|
}
|
||||||
|
|
||||||
func LatoBoldItalic(withSize size: CGFloat) -> UIFont {
|
public func LatoBoldItalic(withSize size: CGFloat) -> UIFont {
|
||||||
UIFont(name: FontName.LatoBoldItalic.rawValue, size: size)!
|
UIFont(name: FontName.LatoBoldItalic.rawValue, size: size)!
|
||||||
}
|
}
|
||||||
|
|
||||||
func LatoLight(withSize size: CGFloat) -> UIFont {
|
public func LatoLight(withSize size: CGFloat) -> UIFont {
|
||||||
UIFont(name: FontName.LatoLight.rawValue, size: size)!
|
UIFont(name: FontName.LatoLight.rawValue, size: size)!
|
||||||
}
|
}
|
||||||
|
|
||||||
func LatoHairlineItalic(withSize size: CGFloat) -> UIFont {
|
public func LatoHairlineItalic(withSize size: CGFloat) -> UIFont {
|
||||||
UIFont(name: FontName.LatoHairlineItalic.rawValue, size: size)!
|
UIFont(name: FontName.LatoHairlineItalic.rawValue, size: size)!
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@@ -6,16 +6,16 @@
|
|||||||
<array/>
|
<array/>
|
||||||
<key>UIAppFonts</key>
|
<key>UIAppFonts</key>
|
||||||
<array>
|
<array>
|
||||||
<string>Lato-Italic</string>
|
<string>Lato-Italic.ttf</string>
|
||||||
<string>Lato-LightItalic</string>
|
<string>Lato-LightItalic.ttf</string>
|
||||||
<string>Lato-Hairline</string>
|
<string>Lato-Thin.ttf</string>
|
||||||
<string>Lato-Bold</string>
|
<string>Lato-Bold.ttf</string>
|
||||||
<string>Lato-Black</string>
|
<string>Lato-Black.ttf</string>
|
||||||
<string>Lato-Regular</string>
|
<string>Lato-Regular.ttf</string>
|
||||||
<string>Lato-BlackItalic</string>
|
<string>Lato-BlackItalic.ttf</string>
|
||||||
<string>Lato-BoldItalic</string>
|
<string>Lato-BoldItalic.ttf</string>
|
||||||
<string>Lato-Light</string>
|
<string>Lato-Light.ttf</string>
|
||||||
<string>Lato-HairlineItalic</string>
|
<string>Lato-ThinItalic.ttf</string>
|
||||||
</array>
|
</array>
|
||||||
</dict>
|
</dict>
|
||||||
</plist>
|
</plist>
|
||||||
|
@@ -6,16 +6,16 @@
|
|||||||
<array/>
|
<array/>
|
||||||
<key>UIAppFonts</key>
|
<key>UIAppFonts</key>
|
||||||
<array>
|
<array>
|
||||||
<string>Lato-Italic</string>
|
<string>Lato-Italic.ttf</string>
|
||||||
<string>Lato-LightItalic</string>
|
<string>Lato-LightItalic.ttf</string>
|
||||||
<string>Lato-Hairline</string>
|
<string>Lato-Thin.ttf</string>
|
||||||
<string>Lato-Bold</string>
|
<string>Lato-Bold.ttf</string>
|
||||||
<string>Lato-Black</string>
|
<string>Lato-Black.ttf</string>
|
||||||
<string>Lato-Regular</string>
|
<string>Lato-Regular.ttf</string>
|
||||||
<string>Lato-BlackItalic</string>
|
<string>Lato-BlackItalic.ttf</string>
|
||||||
<string>Lato-BoldItalic</string>
|
<string>Lato-BoldItalic.ttf</string>
|
||||||
<string>Lato-Light</string>
|
<string>Lato-Light.ttf</string>
|
||||||
<string>Lato-HairlineItalic</string>
|
<string>Lato-ThinItalic.ttf</string>
|
||||||
</array>
|
</array>
|
||||||
</dict>
|
</dict>
|
||||||
</plist>
|
</plist>
|
||||||
|
@@ -1,31 +1,31 @@
|
|||||||
// Generated by ResgenSwift.Images 1.2
|
// Generated by ResgenSwift.Images 2.2.0
|
||||||
// Images from sampleImages
|
// Images from sampleImages
|
||||||
|
|
||||||
import SwiftUI
|
import SwiftUI
|
||||||
|
|
||||||
extension ImageYolo {
|
extension ImageYolo {
|
||||||
|
|
||||||
var article_notification_pull_detail: Image {
|
public var article_notification_pull_detail: Image {
|
||||||
Image("article_notification_pull_detail")
|
Image("article_notification_pull_detail")
|
||||||
}
|
}
|
||||||
|
|
||||||
var article_notification_pull: Image {
|
public var article_notification_pull: Image {
|
||||||
Image("article_notification_pull")
|
Image("article_notification_pull")
|
||||||
}
|
}
|
||||||
|
|
||||||
var new_article: Image {
|
public var new_article: Image {
|
||||||
Image("new_article")
|
Image("new_article")
|
||||||
}
|
}
|
||||||
|
|
||||||
var welcome_background: Image {
|
public var welcome_background: Image {
|
||||||
Image("welcome_background")
|
Image("welcome_background")
|
||||||
}
|
}
|
||||||
|
|
||||||
var article_trash: Image {
|
public var article_trash: Image {
|
||||||
Image("article_trash")
|
Image("article_trash")
|
||||||
}
|
}
|
||||||
|
|
||||||
var ic_close_article: Image {
|
public var ic_close_article: Image {
|
||||||
Image("ic_close_article")
|
Image("ic_close_article")
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
31
SampleFiles/Images/Generated/UIImageYolo+GenAllScript.swift
Normal file
@@ -0,0 +1,31 @@
|
|||||||
|
// Generated by ResgenSwift.Images 2.2.0
|
||||||
|
// Images from sampleImages
|
||||||
|
|
||||||
|
import UIKit
|
||||||
|
|
||||||
|
extension UIImageYolo {
|
||||||
|
|
||||||
|
public var article_notification_pull_detail: UIImage {
|
||||||
|
UIImage(named: "article_notification_pull_detail")!
|
||||||
|
}
|
||||||
|
|
||||||
|
public var article_notification_pull: UIImage {
|
||||||
|
UIImage(named: "article_notification_pull")!
|
||||||
|
}
|
||||||
|
|
||||||
|
public var new_article: UIImage {
|
||||||
|
UIImage(named: "new_article")!
|
||||||
|
}
|
||||||
|
|
||||||
|
public var welcome_background: UIImage {
|
||||||
|
UIImage(named: "welcome_background")!
|
||||||
|
}
|
||||||
|
|
||||||
|
public var article_trash: UIImage {
|
||||||
|
UIImage(named: "article_trash")!
|
||||||
|
}
|
||||||
|
|
||||||
|
public var ic_close_article: UIImage {
|
||||||
|
UIImage(named: "ic_close_article")!
|
||||||
|
}
|
||||||
|
}
|
@@ -1,23 +1,16 @@
|
|||||||
{
|
{
|
||||||
"images" : [
|
"images" : [
|
||||||
{
|
{
|
||||||
"idiom" : "universal",
|
"filename" : "article_notification_pull.svg",
|
||||||
"scale" : "1x",
|
"idiom" : "universal"
|
||||||
"filename" : "article_notification_pull.png"
|
|
||||||
},
|
|
||||||
{
|
|
||||||
"idiom" : "universal",
|
|
||||||
"scale" : "2x",
|
|
||||||
"filename" : "article_notification_pull@2x.png"
|
|
||||||
},
|
|
||||||
{
|
|
||||||
"idiom" : "universal",
|
|
||||||
"scale" : "3x",
|
|
||||||
"filename" : "article_notification_pull@3x.png"
|
|
||||||
}
|
}
|
||||||
],
|
],
|
||||||
"info" : {
|
"info" : {
|
||||||
"version" : 1,
|
"author" : "ResgenSwift-Imagium",
|
||||||
"author" : "ResgenSwift-Imagium"
|
"version" : 1
|
||||||
|
},
|
||||||
|
"properties" : {
|
||||||
|
"preserves-vector-representation" : true,
|
||||||
|
"template-rendering-intent" : "original"
|
||||||
}
|
}
|
||||||
}
|
}
|
Before Width: | Height: | Size: 5.4 KiB |
@@ -0,0 +1,21 @@
|
|||||||
|
<?xml version="1.0" encoding="UTF-8"?>
|
||||||
|
<svg width="111px" height="69px" viewBox="0 0 111 69" version="1.1" xmlns="http://www.w3.org/2000/svg" xmlns:xlink="http://www.w3.org/1999/xlink">
|
||||||
|
<title>article_notification_pull</title>
|
||||||
|
<g id="Page-1" stroke="none" stroke-width="1" fill="none" fill-rule="evenodd">
|
||||||
|
<g id="article_notification_pull">
|
||||||
|
<path d="M81.130223,39.5772 C81.130223,39.5772 99.360223,32.4231 102.116223,33.2806 C104.871223,34.1381 111.204223,51.339 110.013223,51.8526 C108.758223,52.4481 102.001223,47.0974 102.001223,47.0974 C102.001223,47.0974 100.834223,50.2797 98.387023,49.5337 C95.939623,48.7877 92.897623,44.9219 92.897623,44.9219 C92.897623,44.9219 92.488923,47.0487 89.986323,46.4577 C87.483723,45.8667 83.726623,41.567 83.726623,41.567 L81.130223,39.5772 Z" id="Path" fill="#FFFFFF" fill-rule="nonzero"></path>
|
||||||
|
<path d="M99.259223,10.5629 C97.894023,10.5629 96.788623,9.4412 96.788623,8.0557 C96.788623,7.4619 96.333423,7 95.748223,7 C95.163023,7 94.707823,7.4619 94.707823,8.0557 C94.707823,9.4412 93.602423,10.5629 92.237023,10.5629 C91.651723,10.5629 91.196623,11.0247 91.196623,11.6185 C91.196623,12.2124 91.651723,12.6742 92.237023,12.6742 C93.602423,12.6742 94.707823,13.7959 94.707823,15.1814 C94.707823,15.7752 95.163023,16.2371 95.748223,16.2371 C96.333423,16.2371 96.788623,15.7752 96.788623,15.1814 C96.788623,13.7959 97.894023,12.6742 99.259223,12.6742 C99.844223,12.6742 100.300223,12.2124 100.300223,11.6185 C100.365223,11.0907 99.844223,10.5629 99.259223,10.5629 Z" id="Path" fill="#FFFFFF" fill-rule="nonzero"></path>
|
||||||
|
<path d="M81.193523,11.2243 C80.403523,10.8878 79.962423,9.9515 80.333423,9.0805 C80.502023,8.6847 80.287223,8.3125 79.971223,8.1779 C79.576223,8.0096 79.204323,8.2254 79.069423,8.5421 C78.732123,9.3338 77.796523,9.7772 76.927523,9.407 C76.532523,9.2387 76.160623,9.4544 76.025723,9.7711 C75.857023,10.167 76.071823,10.5391 76.387823,10.6738 C77.177923,11.0103 77.619023,11.9466 77.248023,12.8175 C77.079423,13.2134 77.294223,13.5855 77.610223,13.7201 C78.005223,13.8884 78.377123,13.6727 78.512023,13.356 C78.849323,12.5642 79.784923,12.1209 80.653923,12.4911 C81.048923,12.6594 81.420823,12.4436 81.555723,12.1269 C81.769623,11.8439 81.588623,11.3926 81.193523,11.2243 Z" id="Path" fill="#FFFFFF" fill-rule="nonzero"></path>
|
||||||
|
<line x1="12.845223" y1="48.192" x2="16.110123" y2="42" id="Path" stroke-opacity="0.8" stroke="#FFFFFF" stroke-width="2" stroke-linecap="round"></line>
|
||||||
|
<ellipse id="Oval" fill-opacity="0.1" fill="#000000" fill-rule="nonzero" cx="50.345223" cy="67.5" rx="17.5" ry="1.5"></ellipse>
|
||||||
|
<line x1="14.517923" y1="53.5959" x2="17.782723" y2="47.404" id="Path" stroke-opacity="0.8" stroke="#FFFFFF" stroke-width="2" stroke-linecap="round"></line>
|
||||||
|
<line x1="20.806323" y1="52.3897" x2="23.604823" y2="47.0823" id="Path" stroke-opacity="0.8" stroke="#FFFFFF" stroke-width="2" stroke-linecap="round"></line>
|
||||||
|
<path d="M79.876623,1.78144 C79.226423,1.78144 78.641223,1.2536 78.641223,0.52783 C78.641223,0.19794 78.381123,0 78.121023,0 C77.795923,0 77.600823,0.26392 77.600823,0.52783 C77.600823,1.18763 77.080623,1.78144 76.365423,1.78144 C76.040323,1.78144 75.845223,2.04535 75.845223,2.30927 C75.845223,2.63917 76.105323,2.8371 76.365423,2.8371 C77.015623,2.8371 77.600823,3.3649 77.600823,4.0907 C77.600823,4.4206 77.860923,4.6185 78.121023,4.6185 C78.446123,4.6185 78.641223,4.3546 78.641223,4.0907 C78.641223,3.4309 79.161423,2.8371 79.876623,2.8371 C80.201823,2.8371 80.396823,2.57319 80.396823,2.30927 C80.461923,2.04535 80.201823,1.78144 79.876623,1.78144 Z" id="Path" fill="#FFFFFF" fill-rule="nonzero"></path>
|
||||||
|
<path d="M31.021323,19.8983 C31.021323,19.8983 19.126423,4.2537 16.379623,3.4694 C13.568923,2.76686 -0.694914027,14.1215 0.0264229726,15.2225 C0.756502973,16.3967 9.32942297,15.3706 9.32942297,15.3706 C9.32942297,15.3706 8.62499297,18.7201 11.136323,19.3843 C13.647723,20.0484 18.238723,18.3857 18.238723,18.3857 C18.238723,18.3857 17.521923,20.401 19.905523,21.2288 C22.289123,22.0567 27.760723,20.3628 27.760723,20.3628 L31.021323,19.8983 Z" id="Path" fill="#E0EDF3" fill-rule="nonzero"></path>
|
||||||
|
<path d="M89.710223,24.4503 L77.707723,58.391 C77.385623,59.3943 76.267023,59.8992 75.270023,59.5733 L21.900823,40.3589 C20.903823,40.033 20.400423,38.9059 20.722523,37.9026 L32.788923,3.88 C32.835323,3.6518 32.954323,3.4149 33.154823,3.2425 C33.674723,2.66083 34.456323,2.41884 35.299323,2.689 L88.604623,21.9853 C89.374923,22.2641 89.834523,23.0254 89.849423,23.7658 C89.875623,23.9852 89.829323,24.2134 89.710223,24.4503 Z" id="Path" fill="#FFD100" fill-rule="nonzero"></path>
|
||||||
|
<path d="M23.390523,38.0285 L52.772523,31.6916 C54.280623,31.3626 55.821223,31.9203 56.778023,33.1416 L75.925523,57.0459 L57.901923,31.4487 C56.910123,29.9348 55.061323,29.2655 53.361523,29.84 L23.390523,38.0285 Z" id="Path" fill="#DA9135" fill-rule="nonzero"></path>
|
||||||
|
<path d="M89.803023,23.9939 L54.122423,34.35 L33.108423,3.4707 C33.610823,2.74268 34.529023,2.41014 35.299323,2.68899 L88.604623,21.9853 C89.374923,22.2641 89.770623,23.1073 89.803023,23.9939 Z" id="Path" fill="#FFDF4F" fill-rule="nonzero"></path>
|
||||||
|
<path d="M34.370723,5.3975 L52.679723,32.148 C53.572623,33.4511 55.113223,34.0088 56.685323,33.598 L87.402923,24.8749 L56.375623,35.9356 C54.611823,36.592 52.771823,35.9959 51.835123,34.327 L34.370723,5.3975 Z" id="Path" fill="#DA9135" fill-rule="nonzero"></path>
|
||||||
|
</g>
|
||||||
|
</g>
|
||||||
|
</svg>
|
After Width: | Height: | Size: 5.4 KiB |
Before Width: | Height: | Size: 11 KiB |
Before Width: | Height: | Size: 18 KiB |
@@ -1,23 +1,16 @@
|
|||||||
{
|
{
|
||||||
"images" : [
|
"images" : [
|
||||||
{
|
{
|
||||||
"idiom" : "universal",
|
"filename" : "article_notification_pull_detail.svg",
|
||||||
"scale" : "1x",
|
"idiom" : "universal"
|
||||||
"filename" : "article_notification_pull_detail.png"
|
|
||||||
},
|
|
||||||
{
|
|
||||||
"idiom" : "universal",
|
|
||||||
"scale" : "2x",
|
|
||||||
"filename" : "article_notification_pull_detail@2x.png"
|
|
||||||
},
|
|
||||||
{
|
|
||||||
"idiom" : "universal",
|
|
||||||
"scale" : "3x",
|
|
||||||
"filename" : "article_notification_pull_detail@3x.png"
|
|
||||||
}
|
}
|
||||||
],
|
],
|
||||||
"info" : {
|
"info" : {
|
||||||
"version" : 1,
|
"author" : "ResgenSwift-Imagium",
|
||||||
"author" : "ResgenSwift-Imagium"
|
"version" : 1
|
||||||
|
},
|
||||||
|
"properties" : {
|
||||||
|
"preserves-vector-representation" : true,
|
||||||
|
"template-rendering-intent" : "original"
|
||||||
}
|
}
|
||||||
}
|
}
|
Before Width: | Height: | Size: 5.9 KiB |
@@ -0,0 +1,21 @@
|
|||||||
|
<?xml version="1.0" encoding="UTF-8"?>
|
||||||
|
<svg width="111px" height="69px" viewBox="0 0 111 69" version="1.1" xmlns="http://www.w3.org/2000/svg" xmlns:xlink="http://www.w3.org/1999/xlink">
|
||||||
|
<title>article_notification_pull_detail</title>
|
||||||
|
<g id="Page-1" stroke="none" stroke-width="1" fill="none" fill-rule="evenodd">
|
||||||
|
<g id="article_notification_pull_detail" transform="translate(0.000000, 0.000000)">
|
||||||
|
<path d="M81.1301131,39.5772 C81.1301131,39.5772 99.3602131,32.4231 102.116213,33.2806 C104.871213,34.1381 111.204213,51.339 110.013213,51.8526 C108.758213,52.4481 102.001213,47.0974 102.001213,47.0974 C102.001213,47.0974 100.834213,50.2797 98.3869131,49.5337 C95.9395131,48.7877 92.8975131,44.9219 92.8975131,44.9219 C92.8975131,44.9219 92.4888131,47.0487 89.9862131,46.4577 C87.4836131,45.8667 83.7265131,41.567 83.7265131,41.567 L81.1301131,39.5772 Z" id="Path" fill="#E0EDF3" fill-rule="nonzero"></path>
|
||||||
|
<path d="M99.2592131,10.5629 C97.8940131,10.5629 96.7886131,9.4412 96.7886131,8.0557 C96.7886131,7.4619 96.3334131,7 95.7482131,7 C95.1630131,7 94.7078131,7.4619 94.7078131,8.0557 C94.7078131,9.4412 93.6024131,10.5629 92.2370131,10.5629 C91.6517131,10.5629 91.1966131,11.0247 91.1966131,11.6185 C91.1966131,12.2124 91.6517131,12.6742 92.2370131,12.6742 C93.6024131,12.6742 94.7078131,13.7959 94.7078131,15.1814 C94.7078131,15.7752 95.1630131,16.2371 95.7482131,16.2371 C96.3334131,16.2371 96.7886131,15.7752 96.7886131,15.1814 C96.7886131,13.7959 97.8940131,12.6742 99.2592131,12.6742 C99.8442131,12.6742 100.300213,12.2124 100.300213,11.6185 C100.365213,11.0907 99.8442131,10.5629 99.2592131,10.5629 Z" id="Path" fill="#E0EDF3" fill-rule="nonzero"></path>
|
||||||
|
<path d="M81.1934131,11.2243 C80.4034131,10.8878 79.9623131,9.9515 80.3333131,9.0805 C80.5019131,8.6847 80.2871131,8.3125 79.9711131,8.1779 C79.5761131,8.0096 79.2042131,8.2254 79.0693131,8.5421 C78.7320131,9.3338 77.7964131,9.7772 76.9274131,9.407 C76.5324131,9.2387 76.1605131,9.4544 76.0256131,9.7711 C75.8569131,10.167 76.0717131,10.5391 76.3877131,10.6738 C77.1778131,11.0103 77.6189131,11.9466 77.2479131,12.8175 C77.0793131,13.2134 77.2941131,13.5855 77.6101131,13.7201 C78.0051131,13.8884 78.3770131,13.6727 78.5119131,13.356 C78.8492131,12.5642 79.7848131,12.1209 80.6538131,12.4911 C81.0488131,12.6594 81.4207131,12.4436 81.5556131,12.1269 C81.7695131,11.8439 81.5885131,11.3926 81.1934131,11.2243 Z" id="Path" fill="#E0EDF3" fill-rule="nonzero"></path>
|
||||||
|
<line x1="12.8452131" y1="48.192" x2="16.1101131" y2="42" id="Path" stroke-opacity="0.8" stroke="#E0EDF3" stroke-width="2" stroke-linecap="round"></line>
|
||||||
|
<path d="M50.3452131,69 C60.0102131,69 67.8452131,68.3284 67.8452131,67.5 C67.8452131,66.6716 60.0102131,66 50.3452131,66 C40.6802131,66 32.8452131,66.6716 32.8452131,67.5 C32.8452131,68.3284 40.6802131,69 50.3452131,69 Z" id="Path" fill-opacity="0.1" fill="#000000" fill-rule="nonzero"></path>
|
||||||
|
<line x1="14.5178131" y1="53.5959" x2="17.7826131" y2="47.404" id="Path" stroke-opacity="0.8" stroke="#E0EDF3" stroke-width="2" stroke-linecap="round"></line>
|
||||||
|
<line x1="20.8064131" y1="52.3897" x2="23.6049131" y2="47.0823" id="Path" stroke-opacity="0.8" stroke="#E0EDF3" stroke-width="2" stroke-linecap="round"></line>
|
||||||
|
<path d="M79.8766131,1.78144 C79.2264131,1.78144 78.6412131,1.2536 78.6412131,0.52783 C78.6412131,0.19794 78.3811131,0 78.1210131,0 C77.7959131,0 77.6008131,0.26392 77.6008131,0.52783 C77.6008131,1.18763 77.0806131,1.78144 76.3654131,1.78144 C76.0403131,1.78144 75.8452131,2.04535 75.8452131,2.30927 C75.8452131,2.63917 76.1053131,2.8371 76.3654131,2.8371 C77.0156131,2.8371 77.6008131,3.3649 77.6008131,4.0907 C77.6008131,4.4206 77.8609131,4.6185 78.1210131,4.6185 C78.4461131,4.6185 78.6412131,4.3546 78.6412131,4.0907 C78.6412131,3.4309 79.1614131,2.8371 79.8766131,2.8371 C80.2018131,2.8371 80.3968131,2.57319 80.3968131,2.30927 C80.4619131,2.04535 80.2018131,1.78144 79.8766131,1.78144 Z" id="Path" fill="#E0EDF3" fill-rule="nonzero"></path>
|
||||||
|
<path d="M31.0213131,19.8983 C31.0213131,19.8983 19.1264131,4.2537 16.3796131,3.4694 C13.5689131,2.76686 -0.694915869,14.1215 0.0264231309,15.2225 C0.756503131,16.3967 9.32941313,15.3706 9.32941313,15.3706 C9.32941313,15.3706 8.62499313,18.7201 11.1363131,19.3843 C13.6477131,20.0484 18.2387131,18.3857 18.2387131,18.3857 C18.2387131,18.3857 17.5219131,20.401 19.9055131,21.2288 C22.2891131,22.0567 27.7607131,20.3628 27.7607131,20.3628 L31.0213131,19.8983 Z" id="Path" fill="#E0EDF3" fill-rule="nonzero"></path>
|
||||||
|
<path d="M89.7103131,24.4503 L77.7078131,58.391 C77.3857131,59.3943 76.2671131,59.8992 75.2701131,59.5733 L21.9009131,40.3589 C20.9039131,40.033 20.4005131,38.9059 20.7226131,37.9026 L32.7890131,3.88 C32.8354131,3.6518 32.9544131,3.4149 33.1549131,3.2425 C33.6748131,2.66083 34.4564131,2.41884 35.2994131,2.689 L88.6047131,21.9853 C89.3750131,22.2641 89.8346131,23.0254 89.8495131,23.7658 C89.8757131,23.9852 89.8294131,24.2134 89.7103131,24.4503 Z" id="Path" fill="#FFD100" fill-rule="nonzero"></path>
|
||||||
|
<path d="M23.3906131,38.0285 L52.7726131,31.6916 C54.2807131,31.3626 55.8213131,31.9203 56.7781131,33.1416 L75.9256131,57.0459 L57.9020131,31.4487 C56.9102131,29.9348 55.0614131,29.2655 53.3616131,29.84 L23.3906131,38.0285 Z" id="Path" fill="#DA9135" fill-rule="nonzero"></path>
|
||||||
|
<path d="M89.8030131,23.9939 L54.1224131,34.35 L33.1084131,3.4707 C33.6108131,2.74268 34.5290131,2.41014 35.2993131,2.68899 L88.6046131,21.9853 C89.3749131,22.2641 89.7706131,23.1073 89.8030131,23.9939 Z" id="Path" fill="#FFDF4F" fill-rule="nonzero"></path>
|
||||||
|
<path d="M34.3706131,5.3975 L52.6796131,32.148 C53.5725131,33.4511 55.1131131,34.0088 56.6852131,33.598 L87.4028131,24.8749 L56.3755131,35.9356 C54.6117131,36.592 52.7717131,35.9959 51.8350131,34.327 L34.3706131,5.3975 Z" id="Path" fill="#DA9135" fill-rule="nonzero"></path>
|
||||||
|
</g>
|
||||||
|
</g>
|
||||||
|
</svg>
|
After Width: | Height: | Size: 5.8 KiB |
Before Width: | Height: | Size: 12 KiB |
Before Width: | Height: | Size: 19 KiB |
@@ -1,23 +1,16 @@
|
|||||||
{
|
{
|
||||||
"images" : [
|
"images" : [
|
||||||
{
|
{
|
||||||
"idiom" : "universal",
|
"filename" : "article_trash.svg",
|
||||||
"scale" : "1x",
|
"idiom" : "universal"
|
||||||
"filename" : "article_trash.png"
|
|
||||||
},
|
|
||||||
{
|
|
||||||
"idiom" : "universal",
|
|
||||||
"scale" : "2x",
|
|
||||||
"filename" : "article_trash@2x.png"
|
|
||||||
},
|
|
||||||
{
|
|
||||||
"idiom" : "universal",
|
|
||||||
"scale" : "3x",
|
|
||||||
"filename" : "article_trash@3x.png"
|
|
||||||
}
|
}
|
||||||
],
|
],
|
||||||
"info" : {
|
"info" : {
|
||||||
"version" : 1,
|
"author" : "ResgenSwift-Imagium",
|
||||||
"author" : "ResgenSwift-Imagium"
|
"version" : 1
|
||||||
|
},
|
||||||
|
"properties" : {
|
||||||
|
"preserves-vector-representation" : true,
|
||||||
|
"template-rendering-intent" : "original"
|
||||||
}
|
}
|
||||||
}
|
}
|
Before Width: | Height: | Size: 977 B |
4
SampleFiles/Images/imagium.xcassets/article_trash.imageset/article_trash.svg
vendored
Normal file
@@ -0,0 +1,4 @@
|
|||||||
|
<svg width="37" height="41" viewBox="0 0 37 41" fill="none" xmlns="http://www.w3.org/2000/svg">
|
||||||
|
<path d="M1 4.87183L5.45856 32.1227C5.68952 33.5346 6.30709 34.8549 7.24275 35.9371C8.17841 37.0194 9.3956 37.8213 10.7593 38.2539L11.3943 38.4553C15.9682 39.9073 20.8795 39.9073 25.4534 38.4553L26.0884 38.2539C27.4518 37.8215 28.6688 37.0199 29.6044 35.938C30.5401 34.8561 31.1578 33.5362 31.3891 32.1246L35.8476 4.87183" stroke="white" stroke-width="2" stroke-linecap="round" stroke-linejoin="round"/>
|
||||||
|
<path d="M18.4238 8.74392C28.0467 8.74392 35.8476 7.01039 35.8476 4.87196C35.8476 2.73354 28.0467 1 18.4238 1C8.80091 1 1 2.73354 1 4.87196C1 7.01039 8.80091 8.74392 18.4238 8.74392Z" stroke="white" stroke-width="2" stroke-linecap="round" stroke-linejoin="round"/>
|
||||||
|
</svg>
|
After Width: | Height: | Size: 772 B |
Before Width: | Height: | Size: 1.9 KiB |
Before Width: | Height: | Size: 2.9 KiB |
@@ -1,23 +1,16 @@
|
|||||||
{
|
{
|
||||||
"images" : [
|
"images" : [
|
||||||
{
|
{
|
||||||
"idiom" : "universal",
|
"filename" : "ic_close_article.svg",
|
||||||
"scale" : "1x",
|
"idiom" : "universal"
|
||||||
"filename" : "ic_close_article.png"
|
|
||||||
},
|
|
||||||
{
|
|
||||||
"idiom" : "universal",
|
|
||||||
"scale" : "2x",
|
|
||||||
"filename" : "ic_close_article@2x.png"
|
|
||||||
},
|
|
||||||
{
|
|
||||||
"idiom" : "universal",
|
|
||||||
"scale" : "3x",
|
|
||||||
"filename" : "ic_close_article@3x.png"
|
|
||||||
}
|
}
|
||||||
],
|
],
|
||||||
"info" : {
|
"info" : {
|
||||||
"version" : 1,
|
"author" : "ResgenSwift-Imagium",
|
||||||
"author" : "ResgenSwift-Imagium"
|
"version" : 1
|
||||||
|
},
|
||||||
|
"properties" : {
|
||||||
|
"preserves-vector-representation" : true,
|
||||||
|
"template-rendering-intent" : "original"
|
||||||
}
|
}
|
||||||
}
|
}
|
Before Width: | Height: | Size: 330 B |
4
SampleFiles/Images/imagium.xcassets/ic_close_article.imageset/ic_close_article.svg
vendored
Normal file
@@ -0,0 +1,4 @@
|
|||||||
|
<svg width="12" height="12" viewBox="0 0 12 12" fill="none" xmlns="http://www.w3.org/2000/svg">
|
||||||
|
<path d="M11.41 1.41L10 0L5.705 4.295L1.41 0L0 1.41L4.295 5.705L0 10L1.41 11.41L5.705 7.115L10 11.41L11.41 10L7.115 5.705L11.41 1.41Z"
|
||||||
|
fill="#EFF2F4" />
|
||||||
|
</svg>
|
After Width: | Height: | Size: 268 B |
Before Width: | Height: | Size: 513 B |
Before Width: | Height: | Size: 647 B |
@@ -1,23 +1,16 @@
|
|||||||
{
|
{
|
||||||
"images" : [
|
"images" : [
|
||||||
{
|
{
|
||||||
"idiom" : "universal",
|
"filename" : "new_article.svg",
|
||||||
"scale" : "1x",
|
"idiom" : "universal"
|
||||||
"filename" : "new_article.png"
|
|
||||||
},
|
|
||||||
{
|
|
||||||
"idiom" : "universal",
|
|
||||||
"scale" : "2x",
|
|
||||||
"filename" : "new_article@2x.png"
|
|
||||||
},
|
|
||||||
{
|
|
||||||
"idiom" : "universal",
|
|
||||||
"scale" : "3x",
|
|
||||||
"filename" : "new_article@3x.png"
|
|
||||||
}
|
}
|
||||||
],
|
],
|
||||||
"info" : {
|
"info" : {
|
||||||
"version" : 1,
|
"author" : "ResgenSwift-Imagium",
|
||||||
"author" : "ResgenSwift-Imagium"
|
"version" : 1
|
||||||
|
},
|
||||||
|
"properties" : {
|
||||||
|
"preserves-vector-representation" : true,
|
||||||
|
"template-rendering-intent" : "original"
|
||||||
}
|
}
|
||||||
}
|
}
|
Before Width: | Height: | Size: 6.0 KiB |
54
SampleFiles/Images/imagium.xcassets/new_article.imageset/new_article.svg
vendored
Normal file
@@ -0,0 +1,54 @@
|
|||||||
|
<svg width="102" height="69" viewBox="0 0 102 69" fill="none" xmlns="http://www.w3.org/2000/svg">
|
||||||
|
<path d="M100.089 35.765H72.7522V66.626H100.089V35.765Z" fill="#DEF1F8"/>
|
||||||
|
<path d="M89.1215 46.2884H56.9285V66.6806H89.1215V46.2884Z" fill="#FAFAFA"/>
|
||||||
|
<path d="M100.089 46.2883H89.1216V66.6805H100.089V46.2883Z" fill="#DEF1F8"/>
|
||||||
|
<path d="M84.9199 49.5053H82.8464V53.1039H84.9199V49.5053Z" fill="#DEF1F8"/>
|
||||||
|
<path d="M80.8276 49.5053H78.7542V53.1039H80.8276V49.5053Z" fill="#DEF1F8"/>
|
||||||
|
<path d="M62.8759 60.3557H60.8025V63.9543H62.8759V60.3557Z" fill="#DEF1F8"/>
|
||||||
|
<path d="M97.7973 37.7279H95.7239V41.3266H97.7973V37.7279Z" fill="#FAFAFA"/>
|
||||||
|
<path d="M76.9538 37.7279H74.8804V41.3266H76.9538V37.7279Z" fill="#FAFAFA"/>
|
||||||
|
<path d="M36.4457 27.1985H101.509C101.728 25.2457 100.03 23.1303 98.1679 22.4794C96.251 21.8285 94.1699 21.72 92.4721 20.6351C90.3362 19.2791 89.1861 16.5127 86.7216 15.8076C84.3666 15.1566 82.0117 16.7839 79.5472 16.8924C74.5086 17.1094 70.6202 10.8715 65.6364 11.7394C62.2956 12.336 60.3788 15.8076 57.3666 17.3263C54.8473 18.6282 51.8352 18.3569 48.9873 18.7366C44.1678 19.4418 39.2388 23.3472 36.4457 27.1985Z" fill="#DEF1F8"/>
|
||||||
|
<path d="M0.455092 25.2296L32.2995 0.974118C33.9893 -0.324706 36.3288 -0.324706 38.0185 0.974118L69.8305 25.2296C70.1229 25.4569 70.2854 25.8141 70.2854 26.1713V67.7661C70.2854 68.4155 69.7655 68.9351 69.1156 68.9351H1.20246C0.552574 68.9351 0.0326646 68.4155 0.0326646 67.7661V26.1713C0.0001703 25.8141 0.162643 25.4569 0.455092 25.2296Z" fill="url(#paint0_linear)"/>
|
||||||
|
<path d="M53.973 13.183H16.3121L10.5281 17.5666V68.935H59.7245V17.5666L53.973 13.183Z" fill="#FDB918"/>
|
||||||
|
<path d="M55.9878 6.55914H14.2651C12.8353 6.55914 11.6655 7.72808 11.6655 9.15679V64.7789C11.6655 66.2076 12.8353 67.3765 14.2651 67.3765H55.9878C57.4176 67.3765 58.5874 66.2076 58.5874 64.7789V9.15679C58.5874 7.72808 57.4176 6.55914 55.9878 6.55914Z" fill="#FAFAFA"/>
|
||||||
|
<path d="M35.1265 47.2773L68.4657 67.8312C69.2455 68.3182 70.2528 67.7662 70.2528 66.8246V27.6975C70.2528 26.7883 69.2455 26.2039 68.4657 26.6909L35.1265 47.2773Z" fill="url(#paint1_linear)"/>
|
||||||
|
<path d="M0 27.6974V66.8245C0 67.7336 1.00732 68.3181 1.78719 67.831L35.1264 47.2772L1.78719 26.6908C1.00732 26.2038 0 26.7882 0 27.6974Z" fill="url(#paint2_linear)"/>
|
||||||
|
<path d="M40.3255 44.0627L38.1808 42.4392C36.3611 41.0429 33.8591 41.0429 32.0394 42.4392L29.8948 44.0627L35.0939 47.2773L40.3255 44.0627Z" fill="#E6E6E6"/>
|
||||||
|
<path d="M0 67.1491L35.1264 47.2771L29.9273 44.0626L0 67.1491Z" fill="#FDB918"/>
|
||||||
|
<path d="M70.2528 67.1491L35.1265 47.2771L40.3256 44.0626L70.2528 67.1491Z" fill="#FDB918"/>
|
||||||
|
<path d="M29.1147 17.3064C29.1147 17.3064 28.8872 19.969 35.1262 21.4627L33.7939 21.3977C33.7939 21.3977 29.6996 20.9756 28.7573 19.2547C27.8149 17.5337 29.1147 17.3064 29.1147 17.3064Z" fill="#FDB918"/>
|
||||||
|
<path d="M30.1545 15.9104C33.079 13.053 34.9312 21.4304 34.9637 21.4629C35.0287 21.4629 35.0937 21.4629 35.1587 21.4629C35.1587 21.4629 33.014 9.90333 29.4396 14.709C25.6053 19.7744 30.0245 21.3654 35.1262 21.4629C30.7719 21.333 26.8076 19.1574 30.1545 15.9104Z" fill="#FFE368"/>
|
||||||
|
<path d="M40.9753 17.3064C40.9753 17.3064 41.2028 19.969 34.9639 21.4627L36.2961 21.3977C36.2961 21.3977 40.3904 20.9756 41.3328 19.2547C42.2426 17.5337 40.9753 17.3064 40.9753 17.3064Z" fill="#FDB918"/>
|
||||||
|
<path d="M39.9353 15.9104C37.0108 13.053 35.1586 21.4304 35.1261 21.4629C35.0611 21.4629 34.9961 21.4629 34.9312 21.4629C34.9312 21.4629 37.0758 9.90333 40.6502 14.709C44.452 19.7744 40.0328 21.3654 34.9312 21.4629C39.3179 21.333 43.2497 19.1574 39.9353 15.9104Z" fill="#FFE368"/>
|
||||||
|
<path d="M27.3926 36.237V22.0798C27.3926 21.7226 27.685 21.4304 28.0425 21.4304H42.21C42.5674 21.4304 42.8599 21.7226 42.8599 22.0798V36.237C42.8599 36.5941 42.5674 36.8864 42.21 36.8864H28.0425C27.685 36.8864 27.3926 36.5941 27.3926 36.237Z" fill="#063881"/>
|
||||||
|
<path d="M26.3528 24.645V22.0798C26.3528 21.7226 26.6452 21.4304 27.0027 21.4304H43.2499C43.6073 21.4304 43.8997 21.7226 43.8997 22.0798V24.645C43.8997 25.0021 43.6073 25.2944 43.2499 25.2944H27.0027C26.6452 25.2944 26.3528 25.0021 26.3528 24.645Z" fill="#0B559E"/>
|
||||||
|
<path d="M37.5959 21.4304H32.6567V25.2944H37.5959V21.4304Z" fill="#FFE368"/>
|
||||||
|
<path d="M37.3035 25.2943H32.9492V36.8863H37.3035V25.2943Z" fill="#FDB918"/>
|
||||||
|
<path d="M1.5437 69H68.8777C70.1371 69 70.6539 67.4136 69.6204 66.6839L38.0042 44.6961C36.3248 43.427 33.9673 43.427 32.2557 44.6961L0.736338 66.6839C-0.232496 67.4136 0.284216 69 1.5437 69Z" fill="url(#paint3_linear)"/>
|
||||||
|
<defs>
|
||||||
|
<linearGradient id="paint0_linear" x1="-0.00678347" y1="34.4878" x2="70.2659" y2="34.4878" gradientUnits="userSpaceOnUse">
|
||||||
|
<stop stop-color="#FFE368"/>
|
||||||
|
<stop offset="0.3174" stop-color="#FFE265"/>
|
||||||
|
<stop offset="0.5172" stop-color="#FFDD5D"/>
|
||||||
|
<stop offset="0.6845" stop-color="#FED54E"/>
|
||||||
|
<stop offset="0.8339" stop-color="#FECB39"/>
|
||||||
|
<stop offset="0.9701" stop-color="#FDBD1F"/>
|
||||||
|
<stop offset="1" stop-color="#FDB918"/>
|
||||||
|
</linearGradient>
|
||||||
|
<linearGradient id="paint1_linear" x1="9.77743" y1="11.1302" x2="102.999" y2="68.0143" gradientUnits="userSpaceOnUse">
|
||||||
|
<stop stop-color="#FFE368"/>
|
||||||
|
<stop offset="0.3387" stop-color="#FED750"/>
|
||||||
|
<stop offset="1" stop-color="#FDB918"/>
|
||||||
|
</linearGradient>
|
||||||
|
<linearGradient id="paint2_linear" x1="-8.63037" y1="41.2967" x2="84.5914" y2="98.1808" gradientUnits="userSpaceOnUse">
|
||||||
|
<stop stop-color="#FFE368"/>
|
||||||
|
<stop offset="0.3387" stop-color="#FED750"/>
|
||||||
|
<stop offset="1" stop-color="#FDB918"/>
|
||||||
|
</linearGradient>
|
||||||
|
<linearGradient id="paint3_linear" x1="-10.5945" y1="38.7503" x2="87.7585" y2="99.7918" gradientUnits="userSpaceOnUse">
|
||||||
|
<stop stop-color="#FFE368"/>
|
||||||
|
<stop offset="0.3387" stop-color="#FED750"/>
|
||||||
|
<stop offset="1" stop-color="#FDB918"/>
|
||||||
|
</linearGradient>
|
||||||
|
</defs>
|
||||||
|
</svg>
|
After Width: | Height: | Size: 5.5 KiB |
Before Width: | Height: | Size: 13 KiB |
Before Width: | Height: | Size: 20 KiB |
@@ -1,23 +1,23 @@
|
|||||||
{
|
{
|
||||||
"images" : [
|
"images" : [
|
||||||
{
|
{
|
||||||
|
"filename" : "welcome_background.png",
|
||||||
"idiom" : "universal",
|
"idiom" : "universal",
|
||||||
"scale" : "1x",
|
"scale" : "1x"
|
||||||
"filename" : "welcome_background.png"
|
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
|
"filename" : "welcome_background@2x.png",
|
||||||
"idiom" : "universal",
|
"idiom" : "universal",
|
||||||
"scale" : "2x",
|
"scale" : "2x"
|
||||||
"filename" : "welcome_background@2x.png"
|
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
|
"filename" : "welcome_background@3x.png",
|
||||||
"idiom" : "universal",
|
"idiom" : "universal",
|
||||||
"scale" : "3x",
|
"scale" : "3x"
|
||||||
"filename" : "welcome_background@3x.png"
|
|
||||||
}
|
}
|
||||||
],
|
],
|
||||||
"info" : {
|
"info" : {
|
||||||
"version" : 1,
|
"author" : "ResgenSwift-Imagium",
|
||||||
"author" : "ResgenSwift-Imagium"
|
"version" : 1
|
||||||
}
|
}
|
||||||
}
|
}
|
Before Width: | Height: | Size: 281 KiB After Width: | Height: | Size: 281 KiB |
Before Width: | Height: | Size: 813 KiB After Width: | Height: | Size: 813 KiB |
Before Width: | Height: | Size: 1.4 MiB After Width: | Height: | Size: 1.4 MiB |
@@ -1,4 +1,4 @@
|
|||||||
// Generated by ResgenSwift.Strings.Stringium 1.2
|
// Generated by ResgenSwift.Strings.Stringium 2.2.0
|
||||||
|
|
||||||
import UIKit
|
import UIKit
|
||||||
|
|
||||||
@@ -6,7 +6,7 @@ fileprivate let kStringsFileName = "sampleStrings"
|
|||||||
|
|
||||||
extension String {
|
extension String {
|
||||||
|
|
||||||
enum KeyGenAllScript: String {
|
internal enum KeyGenAllScript: String {
|
||||||
case param_lang = "param_lang"
|
case param_lang = "param_lang"
|
||||||
case generic_back = "generic_back"
|
case generic_back = "generic_back"
|
||||||
case generic_loading_data = "generic_loading_data"
|
case generic_loading_data = "generic_loading_data"
|
||||||
@@ -14,7 +14,7 @@ extension String {
|
|||||||
case test_equal_symbol = "test_equal_symbol"
|
case test_equal_symbol = "test_equal_symbol"
|
||||||
case placeholders_test_one = "placeholders_test_one"
|
case placeholders_test_one = "placeholders_test_one"
|
||||||
|
|
||||||
var keyPath: KeyPath<String, String> {
|
internal var keyPath: KeyPath<String, String> {
|
||||||
switch self {
|
switch self {
|
||||||
case .param_lang: return \String.param_lang
|
case .param_lang: return \String.param_lang
|
||||||
case .generic_back: return \String.generic_back
|
case .generic_back: return \String.generic_back
|
||||||
@@ -30,7 +30,10 @@ extension String {
|
|||||||
|
|
||||||
/// Translation in en :
|
/// Translation in en :
|
||||||
/// en
|
/// en
|
||||||
var param_lang: String {
|
///
|
||||||
|
/// Comment :
|
||||||
|
/// No comment
|
||||||
|
internal var param_lang: String {
|
||||||
NSLocalizedString("param_lang", tableName: kStringsFileName, bundle: Bundle.main, value: "en", comment: "")
|
NSLocalizedString("param_lang", tableName: kStringsFileName, bundle: Bundle.main, value: "en", comment: "")
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -38,25 +41,36 @@ extension String {
|
|||||||
|
|
||||||
/// Translation in en :
|
/// Translation in en :
|
||||||
/// Back
|
/// Back
|
||||||
var generic_back: String {
|
///
|
||||||
|
/// Comment :
|
||||||
|
/// No comment
|
||||||
|
internal var generic_back: String {
|
||||||
NSLocalizedString("generic_back", tableName: kStringsFileName, bundle: Bundle.main, value: "Back", comment: "")
|
NSLocalizedString("generic_back", tableName: kStringsFileName, bundle: Bundle.main, value: "Back", comment: "")
|
||||||
}
|
}
|
||||||
|
|
||||||
/// Translation in en :
|
/// Translation in en :
|
||||||
/// Loading data...
|
/// Loading\ndata...
|
||||||
var generic_loading_data: String {
|
///
|
||||||
NSLocalizedString("generic_loading_data", tableName: kStringsFileName, bundle: Bundle.main, value: "Loading data...", comment: "")
|
/// Comment :
|
||||||
|
/// No comment
|
||||||
|
internal var generic_loading_data: String {
|
||||||
|
NSLocalizedString("generic_loading_data", tableName: kStringsFileName, bundle: Bundle.main, value: "Loading\ndata...", comment: "")
|
||||||
}
|
}
|
||||||
|
|
||||||
/// Translation in en :
|
/// Translation in en :
|
||||||
/// Welcome \"%@\" !
|
/// Welcome \"%@\" !
|
||||||
var generic_welcome_firstname_format: String {
|
///
|
||||||
|
/// Comment :
|
||||||
|
/// No comment
|
||||||
|
internal var generic_welcome_firstname_format: String {
|
||||||
NSLocalizedString("generic_welcome_firstname_format", tableName: kStringsFileName, bundle: Bundle.main, value: "Welcome \"%@\" !", comment: "")
|
NSLocalizedString("generic_welcome_firstname_format", tableName: kStringsFileName, bundle: Bundle.main, value: "Welcome \"%@\" !", comment: "")
|
||||||
}
|
}
|
||||||
|
|
||||||
/// Translation in en :
|
/// Translation in en :
|
||||||
/// Welcome \"%@\" !
|
/// Welcome \"%@\" !
|
||||||
func generic_welcome_firstname_format(arg0: String) -> String {
|
///
|
||||||
|
/// Comment :
|
||||||
|
/// No comment
|
||||||
|
internal func generic_welcome_firstname_format(arg0: String) -> String {
|
||||||
String(format: self.generic_welcome_firstname_format, arg0)
|
String(format: self.generic_welcome_firstname_format, arg0)
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -64,7 +78,10 @@ extension String {
|
|||||||
|
|
||||||
/// Translation in en :
|
/// Translation in en :
|
||||||
/// 1€ = 1 point !
|
/// 1€ = 1 point !
|
||||||
var test_equal_symbol: String {
|
///
|
||||||
|
/// Comment :
|
||||||
|
/// No comment
|
||||||
|
internal var test_equal_symbol: String {
|
||||||
NSLocalizedString("test_equal_symbol", tableName: kStringsFileName, bundle: Bundle.main, value: "1€ = 1 point !", comment: "")
|
NSLocalizedString("test_equal_symbol", tableName: kStringsFileName, bundle: Bundle.main, value: "1€ = 1 point !", comment: "")
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -72,13 +89,18 @@ extension String {
|
|||||||
|
|
||||||
/// Translation in en :
|
/// Translation in en :
|
||||||
/// You %%: %2$@ %1$@ Age: %3$d
|
/// You %%: %2$@ %1$@ Age: %3$d
|
||||||
var placeholders_test_one: String {
|
///
|
||||||
|
/// Comment :
|
||||||
|
/// No comment
|
||||||
|
internal var placeholders_test_one: String {
|
||||||
NSLocalizedString("placeholders_test_one", tableName: kStringsFileName, bundle: Bundle.main, value: "You %%: %2$@ %1$@ Age: %3$d", comment: "")
|
NSLocalizedString("placeholders_test_one", tableName: kStringsFileName, bundle: Bundle.main, value: "You %%: %2$@ %1$@ Age: %3$d", comment: "")
|
||||||
}
|
}
|
||||||
|
|
||||||
/// Translation in en :
|
/// Translation in en :
|
||||||
/// You %%: %2$@ %1$@ Age: %3$d
|
/// You %%: %2$@ %1$@ Age: %3$d
|
||||||
func placeholders_test_one(arg0: String, arg1: String, arg2: Int) -> String {
|
///
|
||||||
|
/// Comment :
|
||||||
|
/// No comment
|
||||||
|
internal func placeholders_test_one(arg0: String, arg1: String, arg2: Int) -> String {
|
||||||
String(format: self.placeholders_test_one, arg0, arg1, arg2)
|
String(format: self.placeholders_test_one, arg0, arg1, arg2)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@@ -1,6 +1,6 @@
|
|||||||
/**
|
/**
|
||||||
* Apple Strings File
|
* Apple Strings File
|
||||||
* Generated by ResgenSwift 1.2
|
* Generated by ResgenSwift 2.1.0
|
||||||
* Language: en-us
|
* Language: en-us
|
||||||
*/
|
*/
|
||||||
|
|
||||||
|
@@ -1,6 +1,6 @@
|
|||||||
/**
|
/**
|
||||||
* Apple Strings File
|
* Apple Strings File
|
||||||
* Generated by ResgenSwift 1.2
|
* Generated by ResgenSwift 2.1.0
|
||||||
* Language: en
|
* Language: en
|
||||||
*/
|
*/
|
||||||
|
|
||||||
|
@@ -1,6 +1,6 @@
|
|||||||
/**
|
/**
|
||||||
* Apple Strings File
|
* Apple Strings File
|
||||||
* Generated by ResgenSwift 1.2
|
* Generated by ResgenSwift 2.1.0
|
||||||
* Language: fr
|
* Language: fr
|
||||||
*/
|
*/
|
||||||
|
|
||||||
|
150
SampleFiles/Strings/Generated/sampleStrings.xcstrings
Normal file
@@ -0,0 +1,150 @@
|
|||||||
|
{
|
||||||
|
"sourceLanguage" : "en",
|
||||||
|
"strings" : {
|
||||||
|
"generic_back" : {
|
||||||
|
"comment" : "",
|
||||||
|
"extractionState" : "manual",
|
||||||
|
"localizations" : {
|
||||||
|
"en" : {
|
||||||
|
"stringUnit" : {
|
||||||
|
"state" : "translated",
|
||||||
|
"value" : "Back"
|
||||||
|
}
|
||||||
|
},
|
||||||
|
"en-us" : {
|
||||||
|
"stringUnit" : {
|
||||||
|
"state" : "translated",
|
||||||
|
"value" : "Back"
|
||||||
|
}
|
||||||
|
},
|
||||||
|
"fr" : {
|
||||||
|
"stringUnit" : {
|
||||||
|
"state" : "translated",
|
||||||
|
"value" : "Retour"
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
},
|
||||||
|
"generic_loading_data" : {
|
||||||
|
"comment" : "",
|
||||||
|
"extractionState" : "manual",
|
||||||
|
"localizations" : {
|
||||||
|
"en" : {
|
||||||
|
"stringUnit" : {
|
||||||
|
"state" : "translated",
|
||||||
|
"value" : "Loading\ndata..."
|
||||||
|
}
|
||||||
|
},
|
||||||
|
"en-us" : {
|
||||||
|
"stringUnit" : {
|
||||||
|
"state" : "translated",
|
||||||
|
"value" : "Loading\ndata..."
|
||||||
|
}
|
||||||
|
},
|
||||||
|
"fr" : {
|
||||||
|
"stringUnit" : {
|
||||||
|
"state" : "translated",
|
||||||
|
"value" : "Chargement\ndes données..."
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
},
|
||||||
|
"generic_welcome_firstname_format" : {
|
||||||
|
"comment" : "",
|
||||||
|
"extractionState" : "manual",
|
||||||
|
"localizations" : {
|
||||||
|
"en" : {
|
||||||
|
"stringUnit" : {
|
||||||
|
"state" : "translated",
|
||||||
|
"value" : "Welcome \"%@\" !"
|
||||||
|
}
|
||||||
|
},
|
||||||
|
"en-us" : {
|
||||||
|
"stringUnit" : {
|
||||||
|
"state" : "translated",
|
||||||
|
"value" : "Welcome \"%@\" !"
|
||||||
|
}
|
||||||
|
},
|
||||||
|
"fr" : {
|
||||||
|
"stringUnit" : {
|
||||||
|
"state" : "translated",
|
||||||
|
"value" : "Bienvenue \"%@\" !"
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
},
|
||||||
|
"param_lang" : {
|
||||||
|
"comment" : "",
|
||||||
|
"extractionState" : "manual",
|
||||||
|
"localizations" : {
|
||||||
|
"en" : {
|
||||||
|
"stringUnit" : {
|
||||||
|
"state" : "translated",
|
||||||
|
"value" : "en"
|
||||||
|
}
|
||||||
|
},
|
||||||
|
"en-us" : {
|
||||||
|
"stringUnit" : {
|
||||||
|
"state" : "translated",
|
||||||
|
"value" : "en-us"
|
||||||
|
}
|
||||||
|
},
|
||||||
|
"fr" : {
|
||||||
|
"stringUnit" : {
|
||||||
|
"state" : "translated",
|
||||||
|
"value" : "fr"
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
},
|
||||||
|
"placeholders_test_one" : {
|
||||||
|
"comment" : "",
|
||||||
|
"extractionState" : "manual",
|
||||||
|
"localizations" : {
|
||||||
|
"en" : {
|
||||||
|
"stringUnit" : {
|
||||||
|
"state" : "translated",
|
||||||
|
"value" : "You %%: %2$@ %1$@ Age: %3$d"
|
||||||
|
}
|
||||||
|
},
|
||||||
|
"en-us" : {
|
||||||
|
"stringUnit" : {
|
||||||
|
"state" : "translated",
|
||||||
|
"value" : "You %%: %2$@ %1$@ Age: %3$d"
|
||||||
|
}
|
||||||
|
},
|
||||||
|
"fr" : {
|
||||||
|
"stringUnit" : {
|
||||||
|
"state" : "translated",
|
||||||
|
"value" : "Vous %%: %1$@ %2$@ Age: %3$d"
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
},
|
||||||
|
"test_equal_symbol" : {
|
||||||
|
"comment" : "",
|
||||||
|
"extractionState" : "manual",
|
||||||
|
"localizations" : {
|
||||||
|
"en" : {
|
||||||
|
"stringUnit" : {
|
||||||
|
"state" : "translated",
|
||||||
|
"value" : "1€ = 1 point !"
|
||||||
|
}
|
||||||
|
},
|
||||||
|
"en-us" : {
|
||||||
|
"stringUnit" : {
|
||||||
|
"state" : "translated",
|
||||||
|
"value" : "1€ = 1 point !"
|
||||||
|
}
|
||||||
|
},
|
||||||
|
"fr" : {
|
||||||
|
"stringUnit" : {
|
||||||
|
"state" : "translated",
|
||||||
|
"value" : "1€ = 1 point !"
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
},
|
||||||
|
"version" : "1.0"
|
||||||
|
}
|
@@ -14,11 +14,11 @@
|
|||||||
fr = Retour
|
fr = Retour
|
||||||
en-us = Back
|
en-us = Back
|
||||||
[generic_loading_data]
|
[generic_loading_data]
|
||||||
en = Loading data...
|
en = Loading\ndata...
|
||||||
tags = droid,ios
|
tags = droid,ios
|
||||||
comments =
|
comments =
|
||||||
fr = Chargement des données...
|
fr = Chargement\ndes données...
|
||||||
en-us = Loading data...
|
en-us = Loading\ndata...
|
||||||
[generic_welcome_firstname_format]
|
[generic_welcome_firstname_format]
|
||||||
en = Welcome "%@" !
|
en = Welcome "%@" !
|
||||||
tags = droid,ios
|
tags = droid,ios
|
||||||
|
@@ -1,18 +1,27 @@
|
|||||||
// Generated by ResgenSwift.Analytics 1.2
|
// Generated by ResgenSwift.Analytics 2.2.0
|
||||||
|
|
||||||
|
import Foundation
|
||||||
import MatomoTracker
|
import MatomoTracker
|
||||||
import FirebaseAnalytics
|
import FirebaseAnalytics
|
||||||
|
|
||||||
// MARK: - Protocol
|
// MARK: - Protocol
|
||||||
|
|
||||||
protocol AnalyticsManagerProtocol {
|
public protocol AnalyticsManagerProtocol {
|
||||||
func logScreen(name: String, path: String)
|
|
||||||
|
func logScreen(
|
||||||
|
name: String,
|
||||||
|
path: String,
|
||||||
|
params: [String: Any]?
|
||||||
|
)
|
||||||
|
|
||||||
func logEvent(
|
func logEvent(
|
||||||
name: String,
|
name: String,
|
||||||
action: String,
|
action: String,
|
||||||
category: String,
|
category: String,
|
||||||
params: [String: Any]?
|
params: [String: Any]?
|
||||||
)
|
)
|
||||||
|
|
||||||
|
func setEnable(_ enable: Bool)
|
||||||
}
|
}
|
||||||
|
|
||||||
// MARK: - Matomo
|
// MARK: - Matomo
|
||||||
@@ -47,8 +56,11 @@ class MatomoAnalyticsManager: AnalyticsManagerProtocol {
|
|||||||
|
|
||||||
// MARK: - Methods
|
// MARK: - Methods
|
||||||
|
|
||||||
func logScreen(name: String, path: String) {
|
func logScreen(
|
||||||
guard !tracker.isOptedOut else { return }
|
name: String,
|
||||||
|
path: String,
|
||||||
|
params: [String: Any]?
|
||||||
|
) {
|
||||||
guard let trackerUrl = tracker.contentBase?.absoluteString else { return }
|
guard let trackerUrl = tracker.contentBase?.absoluteString else { return }
|
||||||
|
|
||||||
let urlString = URL(string: "\(trackerUrl)" + "/" + "\(path)" + "iOS")
|
let urlString = URL(string: "\(trackerUrl)" + "/" + "\(path)" + "iOS")
|
||||||
@@ -64,8 +76,6 @@ class MatomoAnalyticsManager: AnalyticsManagerProtocol {
|
|||||||
category: String,
|
category: String,
|
||||||
params: [String: Any]?
|
params: [String: Any]?
|
||||||
) {
|
) {
|
||||||
guard !tracker.isOptedOut else { return }
|
|
||||||
|
|
||||||
tracker.track(
|
tracker.track(
|
||||||
eventWithCategory: category,
|
eventWithCategory: category,
|
||||||
action: action,
|
action: action,
|
||||||
@@ -74,16 +84,43 @@ class MatomoAnalyticsManager: AnalyticsManagerProtocol {
|
|||||||
url: nil
|
url: nil
|
||||||
)
|
)
|
||||||
}
|
}
|
||||||
|
|
||||||
|
func setEnable(_ enable: Bool) {
|
||||||
|
tracker.isOptedOut = !enable
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
// MARK: - Firebase
|
// MARK: - Firebase
|
||||||
|
|
||||||
class FirebaseAnalyticsManager: AnalyticsManagerProtocol {
|
class FirebaseAnalyticsManager: AnalyticsManagerProtocol {
|
||||||
func logScreen(name: String, path: String) {
|
|
||||||
|
// MARK: - Methods
|
||||||
|
|
||||||
|
func logScreen(
|
||||||
|
name: String,
|
||||||
|
path: String,
|
||||||
|
params: [String: Any]?
|
||||||
|
) {
|
||||||
var parameters = [
|
var parameters = [
|
||||||
AnalyticsParameterScreenName: name
|
AnalyticsParameterScreenName: name as NSObject
|
||||||
]
|
]
|
||||||
|
|
||||||
|
if path.isEmpty == false {
|
||||||
|
parameters["path"] = path + "/iOS" as NSObject
|
||||||
|
}
|
||||||
|
|
||||||
|
if let supplementaryParameters = params {
|
||||||
|
for (newKey, newValue) in supplementaryParameters {
|
||||||
|
if parameters.contains(where: { (key: String, value: NSObject) in
|
||||||
|
key == newKey
|
||||||
|
}) {
|
||||||
|
continue
|
||||||
|
}
|
||||||
|
|
||||||
|
parameters[newKey] = newValue as? NSObject
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
Analytics.logEvent(
|
Analytics.logEvent(
|
||||||
AnalyticsEventScreenView,
|
AnalyticsEventScreenView,
|
||||||
parameters: parameters
|
parameters: parameters
|
||||||
@@ -96,56 +133,112 @@ class FirebaseAnalyticsManager: AnalyticsManagerProtocol {
|
|||||||
category: String,
|
category: String,
|
||||||
params: [String: Any]?
|
params: [String: Any]?
|
||||||
) {
|
) {
|
||||||
var parameters: [String:Any] = [
|
var parameters: [String: NSObject] = [
|
||||||
"action": action,
|
AnalyticsParameterItemName: name.replacingOccurrences(of: " ", with: "_") as NSObject
|
||||||
"category": category,
|
|
||||||
]
|
]
|
||||||
|
|
||||||
|
if category.isEmpty == false {
|
||||||
|
parameters["AnalyticsParameterItemCategory"] = category as NSObject
|
||||||
|
}
|
||||||
|
|
||||||
|
if action.isEmpty == false {
|
||||||
|
parameters["action"] = action as NSObject
|
||||||
|
}
|
||||||
|
|
||||||
if let supplementaryParameters = params {
|
if let supplementaryParameters = params {
|
||||||
parameters.merge(supplementaryParameters) { (origin, new) -> Any in
|
for (newKey, newValue) in supplementaryParameters {
|
||||||
return origin
|
if parameters.contains(where: { (key: String, value: NSObject) in
|
||||||
|
key == newKey
|
||||||
|
}) {
|
||||||
|
continue
|
||||||
|
}
|
||||||
|
|
||||||
|
parameters[newKey] = newValue as? NSObject
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
Analytics.logEvent(
|
Analytics.logEvent(
|
||||||
name,
|
AnalyticsEventSelectContent,
|
||||||
parameters: parameters
|
parameters: parameters
|
||||||
)
|
)
|
||||||
}
|
}
|
||||||
|
|
||||||
|
func setEnable(_ enable: Bool) {
|
||||||
|
Analytics.setAnalyticsCollectionEnabled(enable)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// MARK: - Traker Type
|
||||||
|
|
||||||
|
public enum TrackerType: CaseIterable {
|
||||||
|
|
||||||
|
case matomo
|
||||||
|
case firebase
|
||||||
}
|
}
|
||||||
|
|
||||||
// MARK: - Manager
|
// MARK: - Manager
|
||||||
|
|
||||||
class AnalyticsManager {
|
public class AnalyticsManager {
|
||||||
static var shared = AnalyticsManager()
|
|
||||||
|
public static var shared = AnalyticsManager()
|
||||||
|
|
||||||
|
private init() {}
|
||||||
|
|
||||||
// MARK: - Properties
|
// MARK: - Properties
|
||||||
|
|
||||||
var managers: [AnalyticsManagerProtocol] = []
|
var managers: [TrackerType: AnalyticsManagerProtocol] = [:]
|
||||||
|
|
||||||
private var isEnabled: Bool = true
|
private var isEnabled: Bool {
|
||||||
|
if ProcessInfo.processInfo.environment["XCODE_RUNNING_FOR_PREVIEWS"] == "1" {
|
||||||
// MARK: - Methods
|
false
|
||||||
|
} else {
|
||||||
func setAnalyticsEnabled(_ enable: Bool) {
|
true
|
||||||
isEnabled = enable
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
func configure(siteId: String, url: String) {
|
// MARK: - Enable Methods
|
||||||
managers.append(
|
|
||||||
MatomoAnalyticsManager(
|
private func setAnalytics(enable: Bool, _ analytics: [TrackerType]) {
|
||||||
|
managers.forEach { (key, value) in
|
||||||
|
if analytics.contains(where: { type in
|
||||||
|
type == key
|
||||||
|
}) {
|
||||||
|
value.setEnable(enable)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
public func enableAnalytics(_ analytics: [TrackerType] = TrackerType.allCases) {
|
||||||
|
setAnalytics(enable: true, analytics)
|
||||||
|
}
|
||||||
|
|
||||||
|
public func disableAnalytics(_ analytics: [TrackerType] = TrackerType.allCases) {
|
||||||
|
setAnalytics(enable: false, analytics)
|
||||||
|
}
|
||||||
|
|
||||||
|
public func configure(siteId: String, url: String) {
|
||||||
|
managers[TrackerType.matomo] = MatomoAnalyticsManager(
|
||||||
siteId: siteId,
|
siteId: siteId,
|
||||||
url: url
|
url: url
|
||||||
)
|
)
|
||||||
)
|
managers[TrackerType.firebase] = FirebaseAnalyticsManager()
|
||||||
managers.append(FirebaseAnalyticsManager())
|
|
||||||
}
|
}
|
||||||
|
|
||||||
private func logScreen(name: String, path: String) {
|
// MARK: - Private Log Methods
|
||||||
|
|
||||||
|
private func logScreen(
|
||||||
|
name: String,
|
||||||
|
path: String,
|
||||||
|
params: [String: Any]?
|
||||||
|
) {
|
||||||
guard isEnabled else { return }
|
guard isEnabled else { return }
|
||||||
|
|
||||||
managers.forEach { manager in
|
managers.values.forEach { manager in
|
||||||
manager.logScreen(name: name, path: path)
|
manager.logScreen(
|
||||||
|
name: name,
|
||||||
|
path: path,
|
||||||
|
params: params
|
||||||
|
)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -157,7 +250,7 @@ class AnalyticsManager {
|
|||||||
) {
|
) {
|
||||||
guard isEnabled else { return }
|
guard isEnabled else { return }
|
||||||
|
|
||||||
managers.forEach { manager in
|
managers.values.forEach { manager in
|
||||||
manager.logEvent(
|
manager.logEvent(
|
||||||
name: name,
|
name: name,
|
||||||
action: action,
|
action: action,
|
||||||
@@ -169,28 +262,39 @@ class AnalyticsManager {
|
|||||||
|
|
||||||
// MARK: - section_one
|
// MARK: - section_one
|
||||||
|
|
||||||
func logScreenS1DefOne() {
|
public func logScreenS1DefOne(title: String) {
|
||||||
logScreen(
|
logScreen(
|
||||||
name: "s1 def one",
|
name: "s1 def one \(title)",
|
||||||
path: "s1_def_one/"
|
path: "s1_def_one/\(title)",
|
||||||
|
params: nil
|
||||||
)
|
)
|
||||||
}
|
}
|
||||||
|
|
||||||
func logEventS1DefTwo() {
|
public func logEventS1DefTwo(
|
||||||
|
title: String,
|
||||||
|
count: String,
|
||||||
|
test2: String = "test"
|
||||||
|
) {
|
||||||
logEvent(
|
logEvent(
|
||||||
name: "s1 def two",
|
name: "s1 def two",
|
||||||
action: "test",
|
action: "test",
|
||||||
category: "test",
|
category: "test",
|
||||||
params: [:]
|
params: [
|
||||||
|
"title": title,
|
||||||
|
"count": count,
|
||||||
|
"test": "test",
|
||||||
|
"test2": test2
|
||||||
|
]
|
||||||
)
|
)
|
||||||
}
|
}
|
||||||
|
|
||||||
// MARK: - section_two
|
// MARK: - section_two
|
||||||
|
|
||||||
func logScreenS2DefOne() {
|
public func logScreenS2DefOne() {
|
||||||
logScreen(
|
logScreen(
|
||||||
name: "s2 def one",
|
name: "s2 def one",
|
||||||
path: "s2_def_one/"
|
path: "s2_def_one/",
|
||||||
|
params: nil
|
||||||
)
|
)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@@ -1,4 +1,4 @@
|
|||||||
// Generated by ResgenSwift.Strings.Tags 1.2
|
// Generated by ResgenSwift.Strings.Tags 2.2.0
|
||||||
|
|
||||||
import UIKit
|
import UIKit
|
||||||
|
|
||||||
@@ -8,13 +8,19 @@ extension Tags {
|
|||||||
|
|
||||||
/// Translation in ium :
|
/// Translation in ium :
|
||||||
/// Ecran un
|
/// Ecran un
|
||||||
var screen_one: String {
|
///
|
||||||
|
/// Comment :
|
||||||
|
/// No comment
|
||||||
|
public var screen_one: String {
|
||||||
"Ecran un"
|
"Ecran un"
|
||||||
}
|
}
|
||||||
|
|
||||||
/// Translation in ium :
|
/// Translation in ium :
|
||||||
/// Ecran deux
|
/// Ecran deux
|
||||||
var screen_two: String {
|
///
|
||||||
|
/// Comment :
|
||||||
|
/// No comment
|
||||||
|
public var screen_two: String {
|
||||||
"Ecran deux"
|
"Ecran deux"
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@@ -3,9 +3,13 @@ categories:
|
|||||||
- id: section_one
|
- id: section_one
|
||||||
screens:
|
screens:
|
||||||
- id: s1_def_one
|
- id: s1_def_one
|
||||||
name: s1 def one
|
name: s1 def one _TITLE_
|
||||||
path: s1_def_one/
|
path: s1_def_one/_TITLE_
|
||||||
tags: ios
|
tags: ios,droid
|
||||||
|
parameters:
|
||||||
|
- name: title
|
||||||
|
type: String
|
||||||
|
replaceIn: name,path
|
||||||
|
|
||||||
events:
|
events:
|
||||||
- id: s1_def_two
|
- id: s1_def_two
|
||||||
@@ -13,6 +17,17 @@ categories:
|
|||||||
action: test
|
action: test
|
||||||
category: test
|
category: test
|
||||||
tags: ios
|
tags: ios
|
||||||
|
parameters:
|
||||||
|
- name: title
|
||||||
|
type: String
|
||||||
|
- name: count
|
||||||
|
type: String
|
||||||
|
- name: test
|
||||||
|
type: String
|
||||||
|
value: test
|
||||||
|
- name: test2
|
||||||
|
type: String
|
||||||
|
defaultValue: test
|
||||||
|
|
||||||
- id: section_two
|
- id: section_two
|
||||||
screens:
|
screens:
|
||||||
|
@@ -5,66 +5,67 @@ FORCE_FLAG="$1"
|
|||||||
## Font
|
## Font
|
||||||
#swift run -c release ResgenSwift fonts $FORCE_FLAG "./Fonts/sampleFontsAll.txt" \
|
#swift run -c release ResgenSwift fonts $FORCE_FLAG "./Fonts/sampleFontsAll.txt" \
|
||||||
# --extension-output-path "./Fonts/Generated" \
|
# --extension-output-path "./Fonts/Generated" \
|
||||||
# --extension-name "UIFontYolo" \
|
# --extension-name "FontYolo" \
|
||||||
# --extension-name-swift-ui "FontYolo" \
|
# --extension-name-ui-kit "UIFontYolo" \
|
||||||
# --extension-suffix "GenAllScript" \
|
# --extension-suffix "GenAllScript" \
|
||||||
# --info-plist-paths "./Fonts/Generated/test.plist ./Fonts/Generated/test2.plist"
|
# --info-plist-paths "./Fonts/Generated/test.plist ./Fonts/Generated/test2.plist"
|
||||||
#
|
#
|
||||||
#echo "\n-------------------------\n"
|
#echo "\n-------------------------\n"
|
||||||
#
|
#
|
||||||
## Color
|
### Color
|
||||||
#swift run -c release ResgenSwift colors $FORCE_FLAG "./Colors/sampleColors1.txt" \
|
#swift run -c release ResgenSwift colors $FORCE_FLAG "./Colors/sampleColors1.txt" \
|
||||||
# --style all \
|
# --style all \
|
||||||
# --xcassets-path "./Colors/colors.xcassets" \
|
# --xcassets-path "./Colors/colors.xcassets" \
|
||||||
# --extension-output-path "./Colors/Generated/" \
|
# --extension-output-path "./Colors/Generated/" \
|
||||||
# --extension-name "UIColorYolo" \
|
# --extension-name "ColorYolo" \
|
||||||
# --extension-name-swift-ui "ColorYolo" \
|
# --extension-name-ui-kit "UIhkjhkColorYolo" \
|
||||||
# --extension-suffix "GenAllScript"
|
# --extension-suffix "GenAllScript"
|
||||||
#
|
#
|
||||||
#echo "\n-------------------------\n"
|
#echo "\n-------------------------\n"
|
||||||
#
|
#
|
||||||
## Twine
|
### Twine
|
||||||
#swift run -c release ResgenSwift strings twine $FORCE_FLAG "./Twine/sampleStrings.txt" \
|
#swift run -c release ResgenSwift strings twine $FORCE_FLAG "./Twine/sampleStrings.txt" \
|
||||||
# --output-path "./Twine/Generated" \
|
# --output-path "./Twine/Generated" \
|
||||||
# --langs "fr en en-us" \
|
# --langs "fr en en-us" \
|
||||||
# --default-lang "en" \
|
# --default-lang "en" \
|
||||||
# --extension-output-path "./Twine/Generated"
|
# --extension-output-path "./Twine/Generated"
|
||||||
|
#
|
||||||
#echo "\n-------------------------\n"
|
#echo "\n-------------------------\n"
|
||||||
|
|
||||||
## Strings
|
## Strings
|
||||||
#swift run -c release ResgenSwift strings stringium $FORCE_FLAG "./Strings/sampleStrings.txt" \
|
swift run -c release ResgenSwift strings stringium $FORCE_FLAG "./Strings/sampleStrings.txt" \
|
||||||
# --output-path "./Strings/Generated" \
|
--output-path "./Strings/Generated" \
|
||||||
# --langs "fr en en-us" \
|
--langs "fr en en-us" \
|
||||||
# --default-lang "en" \
|
--default-lang "en" \
|
||||||
# --extension-output-path "./Strings/Generated" \
|
--extension-output-path "./Strings/Generated" \
|
||||||
# --extension-name "String" \
|
--extension-name "String" \
|
||||||
# --extension-suffix "GenAllScript"
|
--extension-suffix "GenAllScript"
|
||||||
|
|
||||||
#echo "\n-------------------------\n"
|
#echo "\n-------------------------\n"
|
||||||
|
#
|
||||||
## Tags
|
### Tags
|
||||||
#swift run -c release ResgenSwift strings tags $FORCE_FLAG "./Tags/sampleTags.txt" \
|
#swift run -c release ResgenSwift strings tags $FORCE_FLAG "./Tags/sampleTags.txt" \
|
||||||
# --lang "ium" \
|
# --lang "ium" \
|
||||||
# --extension-output-path "./Tags/Generated" \
|
# --extension-output-path "./Tags/Generated" \
|
||||||
# --extension-name "Tags" \
|
# --extension-name "Tags" \
|
||||||
# --extension-suffix "GenAllScript"
|
# --extension-suffix "GenAllScript"
|
||||||
#
|
#
|
||||||
echo "\n-------------------------\n"
|
#echo "\n-------------------------\n"
|
||||||
|
|
||||||
# Analytics
|
|
||||||
swift run -c release ResgenSwift analytics $FORCE_FLAG "./Tags/sampleTags.yml" \
|
|
||||||
--target "matomo firebase" \
|
|
||||||
--extension-output-path "./Tags/Generated" \
|
|
||||||
--extension-name "Analytics" \
|
|
||||||
--extension-suffix "GenAllScript"
|
|
||||||
|
|
||||||
echo "\n-------------------------\n"
|
|
||||||
#
|
#
|
||||||
## Images
|
## Analytics
|
||||||
|
#swift run -c release ResgenSwift analytics $FORCE_FLAG "./Tags/sampleTags.yml" \
|
||||||
|
# --target "firebase" \
|
||||||
|
# --extension-output-path "./Tags/Generated" \
|
||||||
|
# --extension-name "Analytics" \
|
||||||
|
# --extension-suffix "GenAllScript" \
|
||||||
|
# --static-members true
|
||||||
|
#
|
||||||
|
#echo "\n-------------------------\n"
|
||||||
|
#
|
||||||
|
### Images
|
||||||
#swift run -c release ResgenSwift images $FORCE_FLAG "./Images/sampleImages.txt" \
|
#swift run -c release ResgenSwift images $FORCE_FLAG "./Images/sampleImages.txt" \
|
||||||
# --xcassets-path "./Images/imagium.xcassets" \
|
# --xcassets-path "./Images/imagium.xcassets" \
|
||||||
# --extension-output-path "./Images/Generated" \
|
# --extension-output-path "./Images/Generated" \
|
||||||
# --extension-name "UIImage" \
|
# --extension-name "ImageYolo" \
|
||||||
# --extension-name-swift-ui "ImageYolo" \
|
# --extension-name-ui-kit "UIImageYolo" \
|
||||||
# --extension-suffix "GenAllScript"
|
# --extension-suffix "GenAllScript"
|
||||||
|
@@ -37,6 +37,7 @@ strings:
|
|||||||
extensionOutputPath: ./Strings/Generated
|
extensionOutputPath: ./Strings/Generated
|
||||||
extensionName: String
|
extensionName: String
|
||||||
extensionSuffix: GenAllScript
|
extensionSuffix: GenAllScript
|
||||||
|
visibility: public
|
||||||
|
|
||||||
|
|
||||||
#
|
#
|
||||||
@@ -47,9 +48,10 @@ images:
|
|||||||
inputFile: ./Images/sampleImages.txt
|
inputFile: ./Images/sampleImages.txt
|
||||||
xcassetsPath: ./Images/imagium.xcassets
|
xcassetsPath: ./Images/imagium.xcassets
|
||||||
extensionOutputPath: ./Images/Generated
|
extensionOutputPath: ./Images/Generated
|
||||||
extensionName: UIImage
|
extensionName: ImageYolo
|
||||||
extensionNameSwiftUI: ImageYolo
|
extensionNameUIKit: UIImageYolo
|
||||||
extensionSuffix: GenAllScript
|
extensionSuffix: GenAllScript
|
||||||
|
visibility: public
|
||||||
|
|
||||||
|
|
||||||
#
|
#
|
||||||
@@ -61,9 +63,10 @@ colors:
|
|||||||
style: all
|
style: all
|
||||||
xcassetsPath: ./Colors/colors.xcassets
|
xcassetsPath: ./Colors/colors.xcassets
|
||||||
extensionOutputPath: ./Colors/Generated/
|
extensionOutputPath: ./Colors/Generated/
|
||||||
extensionName: UIColorYolo
|
extensionName: ColorYolo
|
||||||
extensionNameSwiftUI: ColorYolo
|
extensionNameUIKit: UIColorYolo
|
||||||
extensionSuffix: GenAllScript
|
extensionSuffix: GenAllScript
|
||||||
|
visibility: public
|
||||||
|
|
||||||
|
|
||||||
#
|
#
|
||||||
@@ -76,6 +79,7 @@ tags:
|
|||||||
extensionOutputPath: ./Tags/Generated
|
extensionOutputPath: ./Tags/Generated
|
||||||
extensionName: Tags
|
extensionName: Tags
|
||||||
extensionSuffix: GenAllScript
|
extensionSuffix: GenAllScript
|
||||||
|
visibility: public
|
||||||
|
|
||||||
|
|
||||||
#
|
#
|
||||||
@@ -85,9 +89,8 @@ analytics:
|
|||||||
-
|
-
|
||||||
inputFile: ./Tags/sampleTags.yml
|
inputFile: ./Tags/sampleTags.yml
|
||||||
target: "matomo firebase"
|
target: "matomo firebase"
|
||||||
extensionOutputPath: ./Tags/Generated
|
outputFile: ./Tags/Generated/Analytics+GenAllScript.swift
|
||||||
extensionName: Analytics
|
visibility: public
|
||||||
extensionSuffix: GenAllScript
|
|
||||||
|
|
||||||
|
|
||||||
#
|
#
|
||||||
@@ -97,7 +100,8 @@ fonts:
|
|||||||
-
|
-
|
||||||
inputFile: ./Fonts/sampleFontsAll.txt
|
inputFile: ./Fonts/sampleFontsAll.txt
|
||||||
extensionOutputPath: ./Fonts/Generated
|
extensionOutputPath: ./Fonts/Generated
|
||||||
extensionName: UIFontYolo
|
extensionName: FontYolo
|
||||||
extensionNameSwiftUI: FontYolo
|
extensionNameUIKit: UIFontYolo
|
||||||
extensionSuffix: GenAllScript
|
extensionSuffix: GenAllScript
|
||||||
infoPlistPaths: "./Fonts/Generated/test.plist ./Fonts/Generated/test2.plist"
|
infoPlistPaths: "./Fonts/Generated/test.plist ./Fonts/Generated/test2.plist"
|
||||||
|
visibility: public
|
||||||
|
@@ -5,20 +5,19 @@
|
|||||||
// Created by Loris Perret on 08/12/2023.
|
// Created by Loris Perret on 08/12/2023.
|
||||||
//
|
//
|
||||||
|
|
||||||
import ToolCore
|
|
||||||
import Foundation
|
|
||||||
import ArgumentParser
|
import ArgumentParser
|
||||||
|
import Foundation
|
||||||
|
import ToolCore
|
||||||
|
|
||||||
struct Analytics: ParsableCommand {
|
struct Analytics: ParsableCommand {
|
||||||
|
|
||||||
// MARK: - Command Configuration
|
// MARK: - Command Configuration
|
||||||
|
|
||||||
static var configuration = CommandConfiguration(
|
static let configuration = CommandConfiguration(
|
||||||
abstract: "Generate analytics extension file.",
|
abstract: "Generate analytics extension file.",
|
||||||
version: ResgenSwiftVersion
|
version: ResgenSwiftVersion
|
||||||
)
|
)
|
||||||
|
|
||||||
|
|
||||||
// MARK: - Static
|
// MARK: - Static
|
||||||
|
|
||||||
static let toolName = "Analytics"
|
static let toolName = "Analytics"
|
||||||
@@ -33,21 +32,27 @@ struct Analytics: ParsableCommand {
|
|||||||
mutating func run() {
|
mutating func run() {
|
||||||
print("[\(Self.toolName)] Starting analytics generation")
|
print("[\(Self.toolName)] Starting analytics generation")
|
||||||
print("[\(Self.toolName)] Will use inputFile \(options.inputFile) to generate analytics for target: \(options.target)")
|
print("[\(Self.toolName)] Will use inputFile \(options.inputFile) to generate analytics for target: \(options.target)")
|
||||||
|
|
||||||
|
// Check requirements
|
||||||
|
guard checkRequirements() else { return }
|
||||||
|
|
||||||
print("[\(Self.toolName)] Will generate analytics")
|
print("[\(Self.toolName)] Will generate analytics")
|
||||||
|
|
||||||
// Check requirements
|
// Check requirements
|
||||||
guard checkRequirements() else { return }
|
guard checkRequirements() else { return }
|
||||||
|
|
||||||
// Parse input file
|
// Parse input file
|
||||||
let sections = AnalyticsFileParser.parse(options.inputFile, target: options.target)
|
let sections = AnalyticsFileParser().parse(options.inputFile, target: options.target)
|
||||||
|
|
||||||
// Generate extension
|
// Generate extension
|
||||||
AnalyticsGenerator.writeExtensionFiles(sections: sections,
|
AnalyticsGenerator.writeExtensionFiles(
|
||||||
|
sections: sections,
|
||||||
target: options.target,
|
target: options.target,
|
||||||
tags: ["ios", "iosonly"],
|
tags: ["ios", "iosonly"],
|
||||||
staticVar: options.staticMembers,
|
isStatic: options.staticMembers,
|
||||||
extensionName: options.extensionName,
|
outputFile: options.outputFile,
|
||||||
extensionFilePath: options.extensionFilePath)
|
visibility: options.extensionVisibility
|
||||||
|
)
|
||||||
|
|
||||||
print("[\(Self.toolName)] Analytics generated")
|
print("[\(Self.toolName)] Analytics generated")
|
||||||
}
|
}
|
||||||
@@ -61,13 +66,21 @@ struct Analytics: ParsableCommand {
|
|||||||
guard fileManager.fileExists(atPath: options.inputFile) else {
|
guard fileManager.fileExists(atPath: options.inputFile) else {
|
||||||
let error = AnalyticsError.fileNotExists(options.inputFile)
|
let error = AnalyticsError.fileNotExists(options.inputFile)
|
||||||
print(error.description)
|
print(error.description)
|
||||||
Analytics.exit(withError: error)
|
Self.exit(withError: error)
|
||||||
|
}
|
||||||
|
|
||||||
|
guard TrackerType.hasValidTarget(in: options.target) else {
|
||||||
|
let error = AnalyticsError.noValidTracker(options.target)
|
||||||
|
print(error.description)
|
||||||
|
Self.exit(withError: error)
|
||||||
}
|
}
|
||||||
|
|
||||||
// Check if needed to regenerate
|
// Check if needed to regenerate
|
||||||
guard GeneratorChecker.shouldGenerate(force: options.forceGeneration,
|
guard GeneratorChecker.shouldGenerate(
|
||||||
|
force: options.forceGeneration,
|
||||||
inputFilePath: options.inputFile,
|
inputFilePath: options.inputFile,
|
||||||
extensionFilePath: options.extensionFilePath) else {
|
extensionFilePath: options.outputFile
|
||||||
|
) else {
|
||||||
print("[\(Self.toolName)] Analytics are already up to date :) ")
|
print("[\(Self.toolName)] Analytics are already up to date :) ")
|
||||||
return false
|
return false
|
||||||
}
|
}
|
||||||
@@ -75,19 +88,3 @@ struct Analytics: ParsableCommand {
|
|||||||
return true
|
return true
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
extension Analytics {
|
|
||||||
enum TargetType: CaseIterable {
|
|
||||||
case matomo
|
|
||||||
case firebase
|
|
||||||
|
|
||||||
var value: String {
|
|
||||||
switch self {
|
|
||||||
case .matomo:
|
|
||||||
"matomo"
|
|
||||||
case .firebase:
|
|
||||||
"firebase"
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
@@ -8,14 +8,19 @@
|
|||||||
import Foundation
|
import Foundation
|
||||||
|
|
||||||
enum AnalyticsError: Error {
|
enum AnalyticsError: Error {
|
||||||
|
|
||||||
|
case noValidTracker(String)
|
||||||
case fileNotExists(String)
|
case fileNotExists(String)
|
||||||
case missingElement(String)
|
case missingElement(String)
|
||||||
case invalidParameter(String)
|
case invalidParameter(String)
|
||||||
|
case parseFailed(String)
|
||||||
case writeFile(String, String)
|
case writeFile(String, String)
|
||||||
|
|
||||||
var description: String {
|
var description: String {
|
||||||
switch self {
|
switch self {
|
||||||
|
case .noValidTracker(let inputTargets):
|
||||||
|
return "error: [\(Analytics.toolName)] '\(inputTargets)' ne contient aucun tracker valid"
|
||||||
|
|
||||||
case .fileNotExists(let filename):
|
case .fileNotExists(let filename):
|
||||||
return "error: [\(Analytics.toolName)] File \(filename) does not exists"
|
return "error: [\(Analytics.toolName)] File \(filename) does not exists"
|
||||||
|
|
||||||
@@ -25,7 +30,10 @@ enum AnalyticsError: Error {
|
|||||||
case .invalidParameter(let reason):
|
case .invalidParameter(let reason):
|
||||||
return "error: [\(Analytics.toolName)] Invalid parameter \(reason)"
|
return "error: [\(Analytics.toolName)] Invalid parameter \(reason)"
|
||||||
|
|
||||||
case .writeFile(let subErrorDescription, let filename):
|
case .parseFailed(let baseError):
|
||||||
|
return "error: [\(Analytics.toolName)] Parse input file failed: \(baseError)"
|
||||||
|
|
||||||
|
case let .writeFile(subErrorDescription, filename):
|
||||||
return "error: [\(Analytics.toolName)] An error occured while writing content to \(filename): \(subErrorDescription)"
|
return "error: [\(Analytics.toolName)] An error occured while writing content to \(filename): \(subErrorDescription)"
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@@ -5,43 +5,43 @@
|
|||||||
// Created by Loris Perret on 08/12/2023.
|
// Created by Loris Perret on 08/12/2023.
|
||||||
//
|
//
|
||||||
|
|
||||||
import Foundation
|
|
||||||
import ArgumentParser
|
import ArgumentParser
|
||||||
|
import Foundation
|
||||||
|
import ToolCore
|
||||||
|
|
||||||
struct AnalyticsOptions: ParsableArguments {
|
struct AnalyticsOptions: ParsableArguments {
|
||||||
@Flag(name: [.customShort("f"), .customShort("F")], help: "Should force generation")
|
|
||||||
|
@Flag(
|
||||||
|
name: [.customShort("f"), .customShort("F")],
|
||||||
|
help: "Should force generation"
|
||||||
|
)
|
||||||
var forceGeneration = false
|
var forceGeneration = false
|
||||||
|
|
||||||
@Argument(help: "Input files where tags ared defined.", transform: { $0.replaceTiltWithHomeDirectoryPath() })
|
@Argument(
|
||||||
|
help: "Input files where tags ared defined.",
|
||||||
|
transform: { $0.replaceTiltWithHomeDirectoryPath() }
|
||||||
|
)
|
||||||
var inputFile: String
|
var inputFile: String
|
||||||
|
|
||||||
@Option(help: "Target(s) analytics to generate. (\"matomo\" | \"firebase\")")
|
@Option(
|
||||||
|
help: "Target(s) analytics to generate. (\"matomo\" | \"firebase\")",
|
||||||
|
completion: .list(["matotmo", "firebase"])
|
||||||
|
)
|
||||||
var target: String
|
var target: String
|
||||||
|
|
||||||
@Option(help: "Path where to generate the extension.", transform: { $0.replaceTiltWithHomeDirectoryPath() })
|
@Option(
|
||||||
var extensionOutputPath: String
|
help: "Where to generate the analytics manager (path with filename)",
|
||||||
|
transform: { $0.replaceTiltWithHomeDirectoryPath() }
|
||||||
|
)
|
||||||
|
var outputFile: String
|
||||||
|
|
||||||
@Option(help: "Tell if it will generate static properties or not")
|
@Option(help: "Tell if it will generate static properties or not")
|
||||||
var staticMembers: Bool = false
|
var staticMembers: Bool = false
|
||||||
|
|
||||||
@Option(help: "Extension name. If not specified, it will generate a Analytics extension.")
|
@Option(
|
||||||
var extensionName: String = Analytics.defaultExtensionName
|
name: .customLong("visibility"),
|
||||||
|
help: "Visibility of extension and properties. Possibles values: public, private, package, internal. Default is internal",
|
||||||
@Option(help: "Extension suffix. Ex: MyApp, it will generate {extensionName}+Analytics{extensionSuffix}.swift")
|
completion: .list(["public", "private", "package", "internal"])
|
||||||
var extensionSuffix: String?
|
)
|
||||||
}
|
var extensionVisibility: ExtensionVisibility = .internal
|
||||||
|
|
||||||
// MARK: - Computed var
|
|
||||||
|
|
||||||
extension AnalyticsOptions {
|
|
||||||
var extensionFileName: String {
|
|
||||||
if let extensionSuffix = extensionSuffix {
|
|
||||||
return "\(extensionName)+\(extensionSuffix).swift"
|
|
||||||
}
|
|
||||||
return "\(extensionName).swift"
|
|
||||||
}
|
|
||||||
|
|
||||||
var extensionFilePath: String {
|
|
||||||
"\(extensionOutputPath)/\(extensionFileName)"
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
@@ -5,35 +5,54 @@
|
|||||||
// Created by Loris Perret on 08/12/2023.
|
// Created by Loris Perret on 08/12/2023.
|
||||||
//
|
//
|
||||||
|
|
||||||
|
// CPD-OFF
|
||||||
|
|
||||||
|
import CoreVideo
|
||||||
import Foundation
|
import Foundation
|
||||||
import ToolCore
|
import ToolCore
|
||||||
import CoreVideo
|
|
||||||
|
|
||||||
class AnalyticsGenerator {
|
// Disabled cause it's a pain to handle in generated string
|
||||||
static var targets: [Analytics.TargetType] = []
|
// swiftlint:disable type_body_length
|
||||||
|
|
||||||
static func writeExtensionFiles(sections: [AnalyticsCategory], target: String, tags: [String], staticVar: Bool, extensionName: String, extensionFilePath: String) {
|
enum AnalyticsGenerator {
|
||||||
|
|
||||||
|
// MARK: - Write content
|
||||||
|
|
||||||
|
static func writeExtensionFiles(
|
||||||
|
sections: [AnalyticsCategory],
|
||||||
|
target: String,
|
||||||
|
tags: [String],
|
||||||
|
isStatic: Bool,
|
||||||
|
outputFile: String,
|
||||||
|
visibility: ExtensionVisibility
|
||||||
|
) {
|
||||||
// Get target type from enum
|
// Get target type from enum
|
||||||
let targetsString: [String] = target.components(separatedBy: " ")
|
let targetsString: [String] = target.components(separatedBy: " ")
|
||||||
|
let targets = {
|
||||||
Analytics.TargetType.allCases.forEach { enumTarget in
|
var targets = [TrackerType]()
|
||||||
|
TrackerType.allCases.forEach { enumTarget in
|
||||||
if targetsString.contains(enumTarget.value) {
|
if targetsString.contains(enumTarget.value) {
|
||||||
targets.append(enumTarget)
|
targets.append(enumTarget)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
return targets
|
||||||
|
}()
|
||||||
|
|
||||||
// Get extension content
|
// Get extension content
|
||||||
let extensionFileContent = Self.getExtensionContent(sections: sections,
|
let extensionFileContent = getExtensionContent(
|
||||||
|
targets: targets,
|
||||||
|
sections: sections,
|
||||||
tags: tags,
|
tags: tags,
|
||||||
staticVar: staticVar,
|
isStatic: isStatic,
|
||||||
extensionName: extensionName)
|
visibility: visibility
|
||||||
|
)
|
||||||
|
|
||||||
// Write content
|
// Write content
|
||||||
let extensionFilePathURL = URL(fileURLWithPath: extensionFilePath)
|
let outputFilePathURL = URL(fileURLWithPath: outputFile)
|
||||||
do {
|
do {
|
||||||
try extensionFileContent.write(to: extensionFilePathURL, atomically: false, encoding: .utf8)
|
try extensionFileContent.write(to: outputFilePathURL, atomically: false, encoding: .utf8)
|
||||||
} catch (let error) {
|
} catch {
|
||||||
let error = AnalyticsError.writeFile(extensionFilePath, error.localizedDescription)
|
let error = AnalyticsError.writeFile(outputFile, error.localizedDescription)
|
||||||
print(error.description)
|
print(error.description)
|
||||||
Analytics.exit(withError: error)
|
Analytics.exit(withError: error)
|
||||||
}
|
}
|
||||||
@@ -41,60 +60,129 @@ class AnalyticsGenerator {
|
|||||||
|
|
||||||
// MARK: - Extension content
|
// MARK: - Extension content
|
||||||
|
|
||||||
static func getExtensionContent(sections: [AnalyticsCategory], tags: [String], staticVar: Bool, extensionName: String) -> String {
|
static func getExtensionContent(
|
||||||
|
targets: [TrackerType],
|
||||||
|
sections: [AnalyticsCategory],
|
||||||
|
tags: [String],
|
||||||
|
isStatic: Bool,
|
||||||
|
visibility: ExtensionVisibility
|
||||||
|
) -> String {
|
||||||
[
|
[
|
||||||
Self.getHeader(extensionClassname: extensionName, staticVar: staticVar),
|
getHeader(
|
||||||
Self.getProperties(sections: sections, tags: tags, staticVar: staticVar),
|
targets: targets,
|
||||||
Self.getFooter()
|
isStatic: isStatic,
|
||||||
|
visibility: visibility
|
||||||
|
),
|
||||||
|
getProperties(
|
||||||
|
sections: sections,
|
||||||
|
tags: tags,
|
||||||
|
isStatic: isStatic,
|
||||||
|
visibility: visibility
|
||||||
|
),
|
||||||
|
getFooter()
|
||||||
]
|
]
|
||||||
.joined(separator: "\n")
|
.joined(separator: "\n")
|
||||||
}
|
}
|
||||||
|
|
||||||
// MARK: - Extension part
|
// MARK: - Extension part
|
||||||
|
|
||||||
private static func getHeader(extensionClassname: String, staticVar: Bool) -> String {
|
private static func getHeader(
|
||||||
|
targets: [TrackerType],
|
||||||
|
isStatic: Bool,
|
||||||
|
visibility: ExtensionVisibility
|
||||||
|
) -> String {
|
||||||
"""
|
"""
|
||||||
// Generated by ResgenSwift.\(Analytics.toolName) \(ResgenSwiftVersion)
|
// Generated by ResgenSwift.\(Analytics.toolName) \(ResgenSwiftVersion)
|
||||||
|
|
||||||
\(Self.getImport())
|
\(Self.getImport(targets: targets))
|
||||||
|
|
||||||
|
\(Self.getAnalyticsProtocol(targets: targets, visibility: visibility))
|
||||||
|
|
||||||
|
\(Self.getTrackerTypeEnum(targets: targets, visibility: visibility))
|
||||||
|
|
||||||
\(Self.getAnalytics())
|
|
||||||
// MARK: - Manager
|
// MARK: - Manager
|
||||||
|
|
||||||
class AnalyticsManager {
|
\(visibility) class AnalyticsManager {
|
||||||
static var shared = AnalyticsManager()
|
|
||||||
|
\(visibility) static var shared = AnalyticsManager()
|
||||||
|
|
||||||
|
private init() {}
|
||||||
|
|
||||||
// MARK: - Properties
|
// MARK: - Properties
|
||||||
|
|
||||||
var managers: [AnalyticsManagerProtocol] = []
|
var managers: [TrackerType: AnalyticsManagerProtocol] = [:]
|
||||||
|
|
||||||
\(Self.getEnabledContent())
|
\(Self.getEnabledContent(visibility: visibility))
|
||||||
|
|
||||||
\(Self.getAnalyticsProperties())
|
\(Self.getAnalyticsProperties(targets: targets, visibility: visibility))
|
||||||
|
|
||||||
\(Self.getPrivateLogFunction())
|
\(Self.getPrivateLogFunction())
|
||||||
"""
|
"""
|
||||||
}
|
}
|
||||||
|
|
||||||
private static func getEnabledContent() -> String {
|
private static func getTrackerTypeEnum(
|
||||||
"""
|
targets: [TrackerType],
|
||||||
private var isEnabled: Bool = true
|
visibility: ExtensionVisibility
|
||||||
|
) -> String {
|
||||||
|
var result: [String] = []
|
||||||
|
targets.forEach { type in
|
||||||
|
result.append(" case \(type)")
|
||||||
|
}
|
||||||
|
|
||||||
// MARK: - Methods
|
return """
|
||||||
|
// MARK: - Traker Type
|
||||||
|
|
||||||
func setAnalyticsEnabled(_ enable: Bool) {
|
\(visibility) enum TrackerType: CaseIterable {
|
||||||
isEnabled = enable
|
|
||||||
|
\(result.joined(separator: "\n"))
|
||||||
}
|
}
|
||||||
"""
|
"""
|
||||||
}
|
}
|
||||||
|
|
||||||
private static func getImport() -> String {
|
private static func getEnabledContent(
|
||||||
|
visibility: ExtensionVisibility
|
||||||
|
) -> String {
|
||||||
|
"""
|
||||||
|
private var isEnabled: Bool {
|
||||||
|
if ProcessInfo.processInfo.environment["XCODE_RUNNING_FOR_PREVIEWS"] == "1" {
|
||||||
|
false
|
||||||
|
} else {
|
||||||
|
true
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// MARK: - Enable Methods
|
||||||
|
|
||||||
|
private func setAnalytics(enable: Bool, _ analytics: [TrackerType]) {
|
||||||
|
managers.forEach { (key, value) in
|
||||||
|
if analytics.contains(where: { type in
|
||||||
|
type == key
|
||||||
|
}) {
|
||||||
|
value.setEnable(enable)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
\(visibility) func enableAnalytics(_ analytics: [TrackerType] = TrackerType.allCases) {
|
||||||
|
setAnalytics(enable: true, analytics)
|
||||||
|
}
|
||||||
|
|
||||||
|
\(visibility) func disableAnalytics(_ analytics: [TrackerType] = TrackerType.allCases) {
|
||||||
|
setAnalytics(enable: false, analytics)
|
||||||
|
}
|
||||||
|
"""
|
||||||
|
}
|
||||||
|
|
||||||
|
private static func getImport(targets: [TrackerType]) -> String {
|
||||||
var result: [String] = []
|
var result: [String] = []
|
||||||
|
|
||||||
if targets.contains(Analytics.TargetType.matomo) {
|
result.append("import Foundation")
|
||||||
|
|
||||||
|
if targets.contains(TrackerType.matomo) {
|
||||||
result.append("import MatomoTracker")
|
result.append("import MatomoTracker")
|
||||||
}
|
}
|
||||||
if targets.contains(Analytics.TargetType.firebase) {
|
|
||||||
|
if targets.contains(TrackerType.firebase) {
|
||||||
result.append("import FirebaseAnalytics")
|
result.append("import FirebaseAnalytics")
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -103,11 +191,21 @@ class AnalyticsGenerator {
|
|||||||
|
|
||||||
private static func getPrivateLogFunction() -> String {
|
private static func getPrivateLogFunction() -> String {
|
||||||
"""
|
"""
|
||||||
private func logScreen(name: String, path: String) {
|
// MARK: - Private Log Methods
|
||||||
|
|
||||||
|
private func logScreen(
|
||||||
|
name: String,
|
||||||
|
path: String,
|
||||||
|
params: [String: Any]?
|
||||||
|
) {
|
||||||
guard isEnabled else { return }
|
guard isEnabled else { return }
|
||||||
|
|
||||||
managers.forEach { manager in
|
managers.values.forEach { manager in
|
||||||
manager.logScreen(name: name, path: path)
|
manager.logScreen(
|
||||||
|
name: name,
|
||||||
|
path: path,
|
||||||
|
params: params
|
||||||
|
)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -119,7 +217,7 @@ class AnalyticsGenerator {
|
|||||||
) {
|
) {
|
||||||
guard isEnabled else { return }
|
guard isEnabled else { return }
|
||||||
|
|
||||||
managers.forEach { manager in
|
managers.values.forEach { manager in
|
||||||
manager.logEvent(
|
manager.logEvent(
|
||||||
name: name,
|
name: name,
|
||||||
action: action,
|
action: action,
|
||||||
@@ -131,29 +229,31 @@ class AnalyticsGenerator {
|
|||||||
"""
|
"""
|
||||||
}
|
}
|
||||||
|
|
||||||
private static func getAnalyticsProperties() -> String {
|
private static func getAnalyticsProperties(
|
||||||
|
targets: [TrackerType],
|
||||||
|
visibility: ExtensionVisibility
|
||||||
|
) -> String {
|
||||||
var header = ""
|
var header = ""
|
||||||
var content: [String] = []
|
var content: [String] = []
|
||||||
let footer = " }"
|
let footer = " }"
|
||||||
|
|
||||||
if targets.contains(Analytics.TargetType.matomo) {
|
if targets.contains(TrackerType.matomo) {
|
||||||
header = "func configure(siteId: String, url: String) {"
|
header = "\(visibility) func configure(siteId: String, url: String) {"
|
||||||
} else if targets.contains(Analytics.TargetType.firebase) {
|
} else if targets.contains(TrackerType.firebase) {
|
||||||
header = "func configure() {"
|
header = "\(visibility) func configure() {"
|
||||||
}
|
}
|
||||||
|
|
||||||
if targets.contains(Analytics.TargetType.matomo) {
|
if targets.contains(TrackerType.matomo) {
|
||||||
content.append("""
|
content.append("""
|
||||||
managers.append(
|
managers[TrackerType.matomo] = MatomoAnalyticsManager(
|
||||||
MatomoAnalyticsManager(
|
|
||||||
siteId: siteId,
|
siteId: siteId,
|
||||||
url: url
|
url: url
|
||||||
)
|
)
|
||||||
)
|
|
||||||
""")
|
""")
|
||||||
}
|
}
|
||||||
if targets.contains(Analytics.TargetType.firebase) {
|
|
||||||
content.append(" managers.append(FirebaseAnalyticsManager())")
|
if targets.contains(TrackerType.firebase) {
|
||||||
|
content.append(" managers[TrackerType.firebase] = FirebaseAnalyticsManager()")
|
||||||
}
|
}
|
||||||
|
|
||||||
return [
|
return [
|
||||||
@@ -164,36 +264,51 @@ class AnalyticsGenerator {
|
|||||||
.joined(separator: "\n")
|
.joined(separator: "\n")
|
||||||
}
|
}
|
||||||
|
|
||||||
private static func getAnalytics() -> String {
|
private static func getAnalyticsProtocol(
|
||||||
|
targets: [TrackerType],
|
||||||
|
visibility: ExtensionVisibility
|
||||||
|
) -> String {
|
||||||
let proto = """
|
let proto = """
|
||||||
// MARK: - Protocol
|
// MARK: - Protocol
|
||||||
|
|
||||||
protocol AnalyticsManagerProtocol {
|
\(visibility) protocol AnalyticsManagerProtocol {
|
||||||
func logScreen(name: String, path: String)
|
|
||||||
|
func logScreen(
|
||||||
|
name: String,
|
||||||
|
path: String,
|
||||||
|
params: [String: Any]?
|
||||||
|
)
|
||||||
|
|
||||||
func logEvent(
|
func logEvent(
|
||||||
name: String,
|
name: String,
|
||||||
action: String,
|
action: String,
|
||||||
category: String,
|
category: String,
|
||||||
params: [String: Any]?
|
params: [String: Any]?
|
||||||
)
|
)
|
||||||
}
|
|
||||||
|
|
||||||
|
func setEnable(_ enable: Bool)
|
||||||
|
}
|
||||||
"""
|
"""
|
||||||
|
|
||||||
var result: [String] = [proto]
|
var result: [String] = [proto]
|
||||||
|
|
||||||
if targets.contains(Analytics.TargetType.matomo) {
|
if targets.contains(TrackerType.matomo) {
|
||||||
result.append(MatomoGenerator.service.content)
|
result.append(MatomoGenerator.service)
|
||||||
}
|
}
|
||||||
|
|
||||||
if targets.contains(Analytics.TargetType.firebase) {
|
if targets.contains(TrackerType.firebase) {
|
||||||
result.append(FirebaseGenerator.service.content)
|
result.append(FirebaseGenerator.service)
|
||||||
}
|
}
|
||||||
|
|
||||||
return result.joined(separator: "\n")
|
return result.joined(separator: "\n\n")
|
||||||
}
|
}
|
||||||
|
|
||||||
private static func getProperties(sections: [AnalyticsCategory], tags: [String], staticVar: Bool) -> String {
|
private static func getProperties(
|
||||||
|
sections: [AnalyticsCategory],
|
||||||
|
tags: [String],
|
||||||
|
isStatic: Bool,
|
||||||
|
visibility: ExtensionVisibility
|
||||||
|
) -> String {
|
||||||
sections
|
sections
|
||||||
.compactMap { section in
|
.compactMap { section in
|
||||||
// Check that at least one string will be generated
|
// Check that at least one string will be generated
|
||||||
@@ -207,10 +322,10 @@ class AnalyticsGenerator {
|
|||||||
return // Go to next definition
|
return // Go to next definition
|
||||||
}
|
}
|
||||||
|
|
||||||
if staticVar {
|
if isStatic {
|
||||||
res += "\n\n\(definition.getStaticProperty())"
|
res += "\n\n\(definition.getStaticProperty(visibility: visibility))"
|
||||||
} else {
|
} else {
|
||||||
res += "\n\n\(definition.getProperty())"
|
res += "\n\n\(definition.getProperty(visibility: visibility))"
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
return res
|
return res
|
||||||
@@ -225,3 +340,5 @@ class AnalyticsGenerator {
|
|||||||
"""
|
"""
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// CPD-ON
|
||||||
|
@@ -5,36 +5,63 @@
|
|||||||
// Created by Loris Perret on 05/12/2023.
|
// Created by Loris Perret on 05/12/2023.
|
||||||
//
|
//
|
||||||
|
|
||||||
|
// CPD-OFF
|
||||||
|
|
||||||
import Foundation
|
import Foundation
|
||||||
|
|
||||||
enum FirebaseGenerator {
|
enum FirebaseGenerator {
|
||||||
case service
|
|
||||||
|
|
||||||
var content: String {
|
static var service: String {
|
||||||
[
|
[
|
||||||
FirebaseGenerator.service.header,
|
Self.header,
|
||||||
FirebaseGenerator.service.logScreen,
|
Self.logScreen,
|
||||||
FirebaseGenerator.service.logEvent,
|
Self.logEvent,
|
||||||
FirebaseGenerator.service.footer
|
Self.enable,
|
||||||
|
Self.footer
|
||||||
]
|
]
|
||||||
.joined(separator: "\n")
|
.joined(separator: "\n")
|
||||||
}
|
}
|
||||||
|
|
||||||
private var header: String {
|
// MARK: - Private vars
|
||||||
|
|
||||||
|
private static var header: String {
|
||||||
"""
|
"""
|
||||||
// MARK: - Firebase
|
// MARK: - Firebase
|
||||||
|
|
||||||
class FirebaseAnalyticsManager: AnalyticsManagerProtocol {
|
class FirebaseAnalyticsManager: AnalyticsManagerProtocol {
|
||||||
|
|
||||||
|
// MARK: - Methods
|
||||||
|
|
||||||
"""
|
"""
|
||||||
}
|
}
|
||||||
|
|
||||||
private var logScreen: String {
|
private static var logScreen: String {
|
||||||
"""
|
"""
|
||||||
func logScreen(name: String, path: String) {
|
func logScreen(
|
||||||
|
name: String,
|
||||||
|
path: String,
|
||||||
|
params: [String: Any]?
|
||||||
|
) {
|
||||||
var parameters = [
|
var parameters = [
|
||||||
AnalyticsParameterScreenName: name
|
AnalyticsParameterScreenName: name as NSObject
|
||||||
]
|
]
|
||||||
|
|
||||||
|
if path.isEmpty == false {
|
||||||
|
parameters["path"] = path + "/iOS" as NSObject
|
||||||
|
}
|
||||||
|
|
||||||
|
if let supplementaryParameters = params {
|
||||||
|
for (newKey, newValue) in supplementaryParameters {
|
||||||
|
if parameters.contains(where: { (key: String, value: NSObject) in
|
||||||
|
key == newKey
|
||||||
|
}) {
|
||||||
|
continue
|
||||||
|
}
|
||||||
|
|
||||||
|
parameters[newKey] = newValue as? NSObject
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
Analytics.logEvent(
|
Analytics.logEvent(
|
||||||
AnalyticsEventScreenView,
|
AnalyticsEventScreenView,
|
||||||
parameters: parameters
|
parameters: parameters
|
||||||
@@ -44,7 +71,7 @@ enum FirebaseGenerator {
|
|||||||
"""
|
"""
|
||||||
}
|
}
|
||||||
|
|
||||||
private var logEvent: String {
|
private static var logEvent: String {
|
||||||
"""
|
"""
|
||||||
func logEvent(
|
func logEvent(
|
||||||
name: String,
|
name: String,
|
||||||
@@ -52,29 +79,52 @@ enum FirebaseGenerator {
|
|||||||
category: String,
|
category: String,
|
||||||
params: [String: Any]?
|
params: [String: Any]?
|
||||||
) {
|
) {
|
||||||
var parameters: [String:Any] = [
|
var parameters: [String: NSObject] = [
|
||||||
"action": action,
|
AnalyticsParameterItemName: name.replacingOccurrences(of: " ", with: "_") as NSObject
|
||||||
"category": category,
|
|
||||||
]
|
]
|
||||||
|
|
||||||
|
if category.isEmpty == false {
|
||||||
|
parameters["AnalyticsParameterItemCategory"] = category as NSObject
|
||||||
|
}
|
||||||
|
|
||||||
|
if action.isEmpty == false {
|
||||||
|
parameters["action"] = action as NSObject
|
||||||
|
}
|
||||||
|
|
||||||
if let supplementaryParameters = params {
|
if let supplementaryParameters = params {
|
||||||
parameters.merge(supplementaryParameters) { (origin, new) -> Any in
|
for (newKey, newValue) in supplementaryParameters {
|
||||||
return origin
|
if parameters.contains(where: { (key: String, value: NSObject) in
|
||||||
|
key == newKey
|
||||||
|
}) {
|
||||||
|
continue
|
||||||
|
}
|
||||||
|
|
||||||
|
parameters[newKey] = newValue as? NSObject
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
Analytics.logEvent(
|
Analytics.logEvent(
|
||||||
name,
|
AnalyticsEventSelectContent,
|
||||||
parameters: parameters
|
parameters: parameters
|
||||||
)
|
)
|
||||||
}
|
}
|
||||||
"""
|
|
||||||
}
|
|
||||||
|
|
||||||
private var footer: String {
|
|
||||||
"""
|
|
||||||
}
|
|
||||||
|
|
||||||
"""
|
"""
|
||||||
}
|
}
|
||||||
|
|
||||||
|
private static var enable: String {
|
||||||
|
"""
|
||||||
|
func setEnable(_ enable: Bool) {
|
||||||
|
Analytics.setAnalyticsCollectionEnabled(enable)
|
||||||
}
|
}
|
||||||
|
"""
|
||||||
|
}
|
||||||
|
|
||||||
|
private static var footer: String {
|
||||||
|
"""
|
||||||
|
}
|
||||||
|
"""
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// CPD-ON
|
||||||
|
@@ -5,23 +5,27 @@
|
|||||||
// Created by Loris Perret on 05/12/2023.
|
// Created by Loris Perret on 05/12/2023.
|
||||||
//
|
//
|
||||||
|
|
||||||
|
// CPD-OFF
|
||||||
|
|
||||||
import Foundation
|
import Foundation
|
||||||
|
|
||||||
enum MatomoGenerator {
|
enum MatomoGenerator {
|
||||||
case service
|
|
||||||
|
|
||||||
var content: String {
|
static var service: String {
|
||||||
[
|
[
|
||||||
MatomoGenerator.service.header,
|
Self.header,
|
||||||
MatomoGenerator.service.setup,
|
Self.setup,
|
||||||
MatomoGenerator.service.logScreen,
|
Self.logScreen,
|
||||||
MatomoGenerator.service.logEvent,
|
Self.logEvent,
|
||||||
MatomoGenerator.service.footer
|
Self.enable,
|
||||||
|
Self.footer
|
||||||
]
|
]
|
||||||
.joined(separator: "\n")
|
.joined(separator: "\n")
|
||||||
}
|
}
|
||||||
|
|
||||||
private var header: String {
|
// MARK: - Private vars
|
||||||
|
|
||||||
|
private static var header: String {
|
||||||
"""
|
"""
|
||||||
// MARK: - Matomo
|
// MARK: - Matomo
|
||||||
|
|
||||||
@@ -34,7 +38,7 @@ enum MatomoGenerator {
|
|||||||
"""
|
"""
|
||||||
}
|
}
|
||||||
|
|
||||||
private var setup: String {
|
private static var setup: String {
|
||||||
"""
|
"""
|
||||||
// MARK: - Init
|
// MARK: - Init
|
||||||
|
|
||||||
@@ -63,10 +67,13 @@ enum MatomoGenerator {
|
|||||||
"""
|
"""
|
||||||
}
|
}
|
||||||
|
|
||||||
private var logScreen: String {
|
private static var logScreen: String {
|
||||||
"""
|
"""
|
||||||
func logScreen(name: String, path: String) {
|
func logScreen(
|
||||||
guard !tracker.isOptedOut else { return }
|
name: String,
|
||||||
|
path: String,
|
||||||
|
params: [String: Any]?
|
||||||
|
) {
|
||||||
guard let trackerUrl = tracker.contentBase?.absoluteString else { return }
|
guard let trackerUrl = tracker.contentBase?.absoluteString else { return }
|
||||||
|
|
||||||
let urlString = URL(string: "\\(trackerUrl)" + "/" + "\\(path)" + "iOS")
|
let urlString = URL(string: "\\(trackerUrl)" + "/" + "\\(path)" + "iOS")
|
||||||
@@ -79,7 +86,7 @@ enum MatomoGenerator {
|
|||||||
"""
|
"""
|
||||||
}
|
}
|
||||||
|
|
||||||
private var logEvent: String {
|
private static var logEvent: String {
|
||||||
"""
|
"""
|
||||||
func logEvent(
|
func logEvent(
|
||||||
name: String,
|
name: String,
|
||||||
@@ -87,8 +94,6 @@ enum MatomoGenerator {
|
|||||||
category: String,
|
category: String,
|
||||||
params: [String: Any]?
|
params: [String: Any]?
|
||||||
) {
|
) {
|
||||||
guard !tracker.isOptedOut else { return }
|
|
||||||
|
|
||||||
tracker.track(
|
tracker.track(
|
||||||
eventWithCategory: category,
|
eventWithCategory: category,
|
||||||
action: action,
|
action: action,
|
||||||
@@ -97,13 +102,23 @@ enum MatomoGenerator {
|
|||||||
url: nil
|
url: nil
|
||||||
)
|
)
|
||||||
}
|
}
|
||||||
"""
|
|
||||||
}
|
|
||||||
|
|
||||||
private var footer: String {
|
|
||||||
"""
|
|
||||||
}
|
|
||||||
|
|
||||||
"""
|
"""
|
||||||
}
|
}
|
||||||
|
|
||||||
|
private static var enable: String {
|
||||||
|
"""
|
||||||
|
func setEnable(_ enable: Bool) {
|
||||||
|
tracker.isOptedOut = !enable
|
||||||
}
|
}
|
||||||
|
"""
|
||||||
|
}
|
||||||
|
|
||||||
|
private static var footer: String {
|
||||||
|
"""
|
||||||
|
}
|
||||||
|
"""
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// CPD-ON
|
||||||
|
@@ -8,13 +8,20 @@
|
|||||||
import Foundation
|
import Foundation
|
||||||
|
|
||||||
class AnalyticsCategory {
|
class AnalyticsCategory {
|
||||||
|
|
||||||
|
// MARK: - Properties
|
||||||
|
|
||||||
let id: String // OnBoarding
|
let id: String // OnBoarding
|
||||||
var definitions = [AnalyticsDefinition]()
|
var definitions = [AnalyticsDefinition]()
|
||||||
|
|
||||||
|
// MARK: - Init
|
||||||
|
|
||||||
init(id: String) {
|
init(id: String) {
|
||||||
self.id = id
|
self.id = id
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// MARK: - Methods
|
||||||
|
|
||||||
func hasOneOrMoreMatchingTags(tags: [String]) -> Bool {
|
func hasOneOrMoreMatchingTags(tags: [String]) -> Bool {
|
||||||
let allTags = definitions.flatMap { $0.tags }
|
let allTags = definitions.flatMap { $0.tags }
|
||||||
let allTagsSet = Set(allTags)
|
let allTagsSet = Set(allTags)
|
||||||
|
@@ -6,8 +6,12 @@
|
|||||||
//
|
//
|
||||||
|
|
||||||
import Foundation
|
import Foundation
|
||||||
|
import ToolCore
|
||||||
|
|
||||||
class AnalyticsDefinition {
|
class AnalyticsDefinition {
|
||||||
|
|
||||||
|
// MARK: - Properties
|
||||||
|
|
||||||
let id: String
|
let id: String
|
||||||
var name: String
|
var name: String
|
||||||
var path: String = ""
|
var path: String = ""
|
||||||
@@ -18,20 +22,24 @@ class AnalyticsDefinition {
|
|||||||
var parameters: [AnalyticsParameter] = []
|
var parameters: [AnalyticsParameter] = []
|
||||||
var type: TagType
|
var type: TagType
|
||||||
|
|
||||||
|
// MARK: - Init
|
||||||
|
|
||||||
init(id: String, name: String, type: TagType) {
|
init(id: String, name: String, type: TagType) {
|
||||||
self.id = id
|
self.id = id
|
||||||
self.name = name
|
self.name = name
|
||||||
self.type = type
|
self.type = type
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// MARK: - Methods
|
||||||
|
|
||||||
func hasOneOrMoreMatchingTags(inputTags: [String]) -> Bool {
|
func hasOneOrMoreMatchingTags(inputTags: [String]) -> Bool {
|
||||||
if Set(inputTags).intersection(Set(self.tags)).isEmpty {
|
if Set(inputTags).isDisjoint(with: tags) {
|
||||||
return false
|
return false
|
||||||
}
|
}
|
||||||
return true
|
return true
|
||||||
}
|
}
|
||||||
|
|
||||||
// MARK: - Methods
|
// MARK: - Private Methods
|
||||||
|
|
||||||
private func getFuncName() -> String {
|
private func getFuncName() -> String {
|
||||||
var pascalCaseTitle: String = ""
|
var pascalCaseTitle: String = ""
|
||||||
@@ -43,17 +51,25 @@ class AnalyticsDefinition {
|
|||||||
}
|
}
|
||||||
|
|
||||||
private func getParameters() -> String {
|
private func getParameters() -> String {
|
||||||
var params = parameters
|
|
||||||
var result: String
|
var result: String
|
||||||
|
|
||||||
if type == .screen {
|
let paramsString = parameters.compactMap { parameter -> String? in
|
||||||
params = params.filter{ param in
|
guard parameter.value.isEmpty else { return nil }
|
||||||
!param.replaceIn.isEmpty
|
|
||||||
}
|
let defaultValue: String
|
||||||
|
switch parameter.type {
|
||||||
|
case .bool:
|
||||||
|
defaultValue = "\(parameter.defaultValue.lowercased())"
|
||||||
|
|
||||||
|
case .int, .double:
|
||||||
|
defaultValue = "\(parameter.defaultValue)"
|
||||||
|
|
||||||
|
case .string:
|
||||||
|
defaultValue = "\"\(parameter.defaultValue)\""
|
||||||
}
|
}
|
||||||
|
|
||||||
let paramsString = params.map { parameter in
|
let defaultValueString = parameter.defaultValue.isEmpty ? "" : " = \(defaultValue)"
|
||||||
"\(parameter.name): \(parameter.type)"
|
return "\(parameter.name): \(parameter.type.rawValue)\(defaultValueString)"
|
||||||
}
|
}
|
||||||
|
|
||||||
if paramsString.count > 2 {
|
if paramsString.count > 2 {
|
||||||
@@ -76,10 +92,17 @@ class AnalyticsDefinition {
|
|||||||
for rep in parameter.replaceIn {
|
for rep in parameter.replaceIn {
|
||||||
switch rep {
|
switch rep {
|
||||||
case "name": name = name.replacingFirstOccurrence(of: "_\(parameter.name.uppercased())_", with: "\\(\(parameter.name))")
|
case "name": name = name.replacingFirstOccurrence(of: "_\(parameter.name.uppercased())_", with: "\\(\(parameter.name))")
|
||||||
|
|
||||||
case "path": path = path.replacingFirstOccurrence(of: "_\(parameter.name.uppercased())_", with: "\\(\(parameter.name))")
|
case "path": path = path.replacingFirstOccurrence(of: "_\(parameter.name.uppercased())_", with: "\\(\(parameter.name))")
|
||||||
|
|
||||||
case "category": category = category.replacingFirstOccurrence(of: "_\(parameter.name.uppercased())_", with: "\\(\(parameter.name))")
|
case "category": category = category.replacingFirstOccurrence(of: "_\(parameter.name.uppercased())_", with: "\\(\(parameter.name))")
|
||||||
|
|
||||||
case "action": action = action.replacingFirstOccurrence(of: "_\(parameter.name.uppercased())_", with: "\\(\(parameter.name))")
|
case "action": action = action.replacingFirstOccurrence(of: "_\(parameter.name.uppercased())_", with: "\\(\(parameter.name))")
|
||||||
default: break
|
|
||||||
|
default:
|
||||||
|
if let param = parameters.first(where: { $0.name == rep }), param.value.isEmpty == false {
|
||||||
|
param.value = param.value.replacingFirstOccurrence(of: "_\(parameter.name.uppercased())_", with: "\\(\(parameter.name))")
|
||||||
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@@ -94,7 +117,20 @@ class AnalyticsDefinition {
|
|||||||
}
|
}
|
||||||
|
|
||||||
supplementaryParams.forEach { param in
|
supplementaryParams.forEach { param in
|
||||||
|
if param.value.isEmpty {
|
||||||
params.append("\"\(param.name)\": \(param.name)")
|
params.append("\"\(param.name)\": \(param.name)")
|
||||||
|
} else {
|
||||||
|
switch param.type {
|
||||||
|
case .bool:
|
||||||
|
params.append("\"\(param.name)\": \(param.value.lowercased())")
|
||||||
|
|
||||||
|
case .int, .double:
|
||||||
|
params.append("\"\(param.name)\": \(param.value)")
|
||||||
|
|
||||||
|
case .string:
|
||||||
|
params.append("\"\(param.name)\": \"\(param.value)\"")
|
||||||
|
}
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
if params.count > 1 {
|
if params.count > 1 {
|
||||||
@@ -108,14 +144,15 @@ class AnalyticsDefinition {
|
|||||||
[\(params.joined(separator: ", "))]
|
[\(params.joined(separator: ", "))]
|
||||||
"""
|
"""
|
||||||
} else {
|
} else {
|
||||||
result = "[:]"
|
result = "nil"
|
||||||
}
|
}
|
||||||
|
|
||||||
if type == .screen {
|
if type == .screen {
|
||||||
return """
|
return """
|
||||||
logScreen(
|
logScreen(
|
||||||
name: "\(name)",
|
name: "\(name)",
|
||||||
path: "\(path)"
|
path: "\(path)",
|
||||||
|
params: \(result)
|
||||||
)
|
)
|
||||||
"""
|
"""
|
||||||
} else {
|
} else {
|
||||||
@@ -132,45 +169,21 @@ class AnalyticsDefinition {
|
|||||||
|
|
||||||
// MARK: - Raw strings
|
// MARK: - Raw strings
|
||||||
|
|
||||||
func getProperty() -> String {
|
func getProperty(visibility: ExtensionVisibility) -> String {
|
||||||
replaceIn()
|
replaceIn()
|
||||||
return """
|
return """
|
||||||
func \(getFuncName())\(getParameters()) {
|
\(visibility) func \(getFuncName())\(getParameters()) {
|
||||||
\(getlogFunction())
|
\(getlogFunction())
|
||||||
}
|
}
|
||||||
"""
|
"""
|
||||||
}
|
}
|
||||||
|
|
||||||
func getStaticProperty() -> String {
|
func getStaticProperty(visibility: ExtensionVisibility) -> String {
|
||||||
replaceIn()
|
replaceIn()
|
||||||
return """
|
return """
|
||||||
static func \(getFuncName())\(getParameters()) {
|
\(visibility) static func \(getFuncName())\(getParameters()) {
|
||||||
\(getlogFunction())
|
AnalyticsManager.shared.\(getlogFunction())
|
||||||
}
|
}
|
||||||
"""
|
"""
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
extension AnalyticsDefinition {
|
|
||||||
enum TagType {
|
|
||||||
case screen
|
|
||||||
case event
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
extension String {
|
|
||||||
func replacingFirstOccurrence(of: String, with: String) -> Self {
|
|
||||||
if let range = self.range(of: of) {
|
|
||||||
let tmp = self.replacingOccurrences(
|
|
||||||
of: of,
|
|
||||||
with: with,
|
|
||||||
options: .literal,
|
|
||||||
range: range
|
|
||||||
)
|
|
||||||
|
|
||||||
return tmp
|
|
||||||
}
|
|
||||||
|
|
||||||
return self
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
@@ -8,18 +8,19 @@
|
|||||||
import Foundation
|
import Foundation
|
||||||
|
|
||||||
struct AnalyticsFile: Codable {
|
struct AnalyticsFile: Codable {
|
||||||
|
|
||||||
var categories: [AnalyticsCategoryDTO]
|
var categories: [AnalyticsCategoryDTO]
|
||||||
}
|
}
|
||||||
|
|
||||||
struct AnalyticsCategoryDTO: Codable {
|
struct AnalyticsCategoryDTO: Codable {
|
||||||
|
|
||||||
var id: String
|
var id: String
|
||||||
var screens: [AnalyticsDefinitionScreenDTO]?
|
var screens: [AnalyticsDefinitionScreenDTO]?
|
||||||
var events: [AnalyticsDefinitionEventDTO]?
|
var events: [AnalyticsDefinitionEventDTO]?
|
||||||
}
|
}
|
||||||
|
|
||||||
protocol AnalyticsDefinitionDTO: Codable {}
|
struct AnalyticsDefinitionScreenDTO: Codable {
|
||||||
|
|
||||||
struct AnalyticsDefinitionScreenDTO: AnalyticsDefinitionDTO {
|
|
||||||
var id: String
|
var id: String
|
||||||
var name: String
|
var name: String
|
||||||
var tags: String
|
var tags: String
|
||||||
@@ -29,7 +30,8 @@ struct AnalyticsDefinitionScreenDTO: AnalyticsDefinitionDTO {
|
|||||||
var path: String?
|
var path: String?
|
||||||
}
|
}
|
||||||
|
|
||||||
struct AnalyticsDefinitionEventDTO: AnalyticsDefinitionDTO {
|
struct AnalyticsDefinitionEventDTO: Codable {
|
||||||
|
|
||||||
var id: String
|
var id: String
|
||||||
var name: String
|
var name: String
|
||||||
var tags: String
|
var tags: String
|
||||||
@@ -41,7 +43,10 @@ struct AnalyticsDefinitionEventDTO: AnalyticsDefinitionDTO {
|
|||||||
}
|
}
|
||||||
|
|
||||||
struct AnalyticsParameterDTO: Codable {
|
struct AnalyticsParameterDTO: Codable {
|
||||||
|
|
||||||
var name: String
|
var name: String
|
||||||
var type: String
|
var type: String
|
||||||
|
var value: String?
|
||||||
|
var defaultValue: String?
|
||||||
var replaceIn: String?
|
var replaceIn: String?
|
||||||
}
|
}
|
||||||
|
@@ -8,12 +8,21 @@
|
|||||||
import Foundation
|
import Foundation
|
||||||
|
|
||||||
class AnalyticsParameter {
|
class AnalyticsParameter {
|
||||||
|
|
||||||
|
// MARK: - Properties
|
||||||
|
|
||||||
var name: String
|
var name: String
|
||||||
var type: String
|
var type: ParameterType
|
||||||
|
var value: String
|
||||||
|
var defaultValue: String
|
||||||
var replaceIn: [String] = []
|
var replaceIn: [String] = []
|
||||||
|
|
||||||
init(name: String, type: String) {
|
// MARK: - Init
|
||||||
|
|
||||||
|
init(name: String, type: ParameterType, value: String, defaultValue: String) {
|
||||||
self.name = name
|
self.name = name
|
||||||
self.type = type
|
self.type = type
|
||||||
|
self.value = value
|
||||||
|
self.defaultValue = defaultValue
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
16
Sources/ResgenSwift/Analytics/Model/ParameterType.swift
Normal file
@@ -0,0 +1,16 @@
|
|||||||
|
//
|
||||||
|
// File.swift
|
||||||
|
//
|
||||||
|
//
|
||||||
|
// Created by Loris Perret on 17/07/2024.
|
||||||
|
//
|
||||||
|
|
||||||
|
import Foundation
|
||||||
|
|
||||||
|
enum ParameterType: String {
|
||||||
|
|
||||||
|
case string = "String"
|
||||||
|
case int = "Int"
|
||||||
|
case double = "Double"
|
||||||
|
case bool = "Bool"
|
||||||
|
}
|
17
Sources/ResgenSwift/Analytics/Model/TagType.swift
Normal file
@@ -0,0 +1,17 @@
|
|||||||
|
//
|
||||||
|
// TagType.swift
|
||||||
|
//
|
||||||
|
//
|
||||||
|
// Created by Thibaut Schmitt on 08/12/2023.
|
||||||
|
//
|
||||||
|
|
||||||
|
import Foundation
|
||||||
|
|
||||||
|
extension AnalyticsDefinition {
|
||||||
|
|
||||||
|
enum TagType {
|
||||||
|
|
||||||
|
case screen
|
||||||
|
case event
|
||||||
|
}
|
||||||
|
}
|
31
Sources/ResgenSwift/Analytics/Model/TargetType.swift
Normal file
@@ -0,0 +1,31 @@
|
|||||||
|
//
|
||||||
|
// TargetType.swift
|
||||||
|
//
|
||||||
|
//
|
||||||
|
// Created by Thibaut Schmitt on 08/12/2023.
|
||||||
|
//
|
||||||
|
|
||||||
|
import Foundation
|
||||||
|
|
||||||
|
enum TrackerType: CaseIterable, Sendable {
|
||||||
|
|
||||||
|
case matomo
|
||||||
|
case firebase
|
||||||
|
|
||||||
|
var value: String {
|
||||||
|
switch self {
|
||||||
|
case .matomo:
|
||||||
|
"matomo"
|
||||||
|
|
||||||
|
case .firebase:
|
||||||
|
"firebase"
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
static func hasValidTarget(in targets: String) -> Bool {
|
||||||
|
for tracker in Self.allCases where targets.contains(tracker.value) {
|
||||||
|
return true
|
||||||
|
}
|
||||||
|
return false
|
||||||
|
}
|
||||||
|
}
|
@@ -9,54 +9,130 @@ import Foundation
|
|||||||
import Yams
|
import Yams
|
||||||
|
|
||||||
class AnalyticsFileParser {
|
class AnalyticsFileParser {
|
||||||
private static var inputFile: String = ""
|
|
||||||
private static var target: String = ""
|
|
||||||
|
|
||||||
private static func parseYaml() -> AnalyticsFile {
|
// MARK: - Properties
|
||||||
|
|
||||||
|
private var inputFile: String = ""
|
||||||
|
private var target: String = ""
|
||||||
|
|
||||||
|
// MARK: - Methods
|
||||||
|
|
||||||
|
func parse(_ inputFile: String, target: String) -> [AnalyticsCategory] {
|
||||||
|
self.inputFile = inputFile
|
||||||
|
self.target = target
|
||||||
|
|
||||||
|
let tagFile = parseYaml()
|
||||||
|
|
||||||
|
return tagFile
|
||||||
|
.categories
|
||||||
|
.map { categorie in
|
||||||
|
let section = AnalyticsCategory(id: categorie.id)
|
||||||
|
|
||||||
|
if let screens = categorie.screens {
|
||||||
|
section
|
||||||
|
.definitions
|
||||||
|
.append(
|
||||||
|
contentsOf: getTagDefinitionScreen(from: screens)
|
||||||
|
)
|
||||||
|
}
|
||||||
|
|
||||||
|
if let events = categorie.events {
|
||||||
|
section
|
||||||
|
.definitions
|
||||||
|
.append(
|
||||||
|
contentsOf: getTagDefinitionEvent(from: events)
|
||||||
|
)
|
||||||
|
}
|
||||||
|
|
||||||
|
return section
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// MARK: - Private methods
|
||||||
|
|
||||||
|
private func parseYaml() -> AnalyticsFile {
|
||||||
guard let data = FileManager().contents(atPath: inputFile) else {
|
guard let data = FileManager().contents(atPath: inputFile) else {
|
||||||
let error = AnalyticsError.fileNotExists(inputFile)
|
let error = AnalyticsError.fileNotExists(inputFile)
|
||||||
|
print(error.description)
|
||||||
Analytics.exit(withError: error)
|
Analytics.exit(withError: error)
|
||||||
}
|
}
|
||||||
|
|
||||||
do {
|
do {
|
||||||
let tagFile = try YAMLDecoder().decode(AnalyticsFile.self, from: data)
|
let tagFile = try YAMLDecoder().decode(AnalyticsFile.self, from: data)
|
||||||
return tagFile
|
return tagFile
|
||||||
} catch let error {
|
} catch {
|
||||||
|
let error = AnalyticsError.parseFailed(error.localizedDescription)
|
||||||
|
print(error.description)
|
||||||
Analytics.exit(withError: error)
|
Analytics.exit(withError: error)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
private static func getParameters(fromData data: [AnalyticsParameterDTO]) -> [AnalyticsParameter] {
|
private static func getParameters(from parameters: [AnalyticsParameterDTO]) -> [AnalyticsParameter] {
|
||||||
var parameters: [AnalyticsParameter] = []
|
func verify(value: String?, for type: ParameterType) {
|
||||||
|
guard let value, value.isEmpty == false else { return }
|
||||||
|
|
||||||
data.forEach { value in
|
switch type {
|
||||||
|
case .int:
|
||||||
|
if Int(value) == nil {
|
||||||
|
let error = AnalyticsError.invalidParameter("type of \(value) is not \(type)")
|
||||||
|
print(error.description)
|
||||||
|
Analytics.exit(withError: error)
|
||||||
|
}
|
||||||
|
|
||||||
|
case .bool:
|
||||||
|
if Bool(value.lowercased()) == nil {
|
||||||
|
let error = AnalyticsError.invalidParameter("type of \(value) is not \(type)")
|
||||||
|
print(error.description)
|
||||||
|
Analytics.exit(withError: error)
|
||||||
|
}
|
||||||
|
|
||||||
|
case .double:
|
||||||
|
if Double(value) == nil {
|
||||||
|
let error = AnalyticsError.invalidParameter("type of \(value) is not \(type)")
|
||||||
|
print(error.description)
|
||||||
|
Analytics.exit(withError: error)
|
||||||
|
}
|
||||||
|
|
||||||
|
case .string:
|
||||||
|
break
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return parameters.map { dtoParameter in
|
||||||
// Type
|
// Type
|
||||||
|
let type = dtoParameter.type.uppercasedFirst()
|
||||||
|
|
||||||
let type = value.type.uppercasedFirst()
|
guard let typeEnum = ParameterType(rawValue: type) else {
|
||||||
|
let error = AnalyticsError.invalidParameter("type of \(dtoParameter.name)")
|
||||||
guard
|
print(error.description)
|
||||||
type == "String" ||
|
|
||||||
type == "Int" ||
|
|
||||||
type == "Double" ||
|
|
||||||
type == "Bool"
|
|
||||||
else {
|
|
||||||
let error = AnalyticsError.invalidParameter("type of \(value.name)")
|
|
||||||
Analytics.exit(withError: error)
|
Analytics.exit(withError: error)
|
||||||
}
|
}
|
||||||
|
|
||||||
let parameter: AnalyticsParameter = AnalyticsParameter(name: value.name, type: type)
|
if dtoParameter.value != nil, dtoParameter.replaceIn != nil {
|
||||||
|
let error = AnalyticsError.invalidParameter("you can't set 'value' and 'replaceIn' for \(dtoParameter.name)")
|
||||||
|
print(error.description)
|
||||||
|
Analytics.exit(withError: error)
|
||||||
|
}
|
||||||
|
|
||||||
if let replaceIn = value.replaceIn {
|
verify(value: dtoParameter.value, for: typeEnum)
|
||||||
|
verify(value: dtoParameter.defaultValue, for: typeEnum)
|
||||||
|
|
||||||
|
let parameter = AnalyticsParameter(
|
||||||
|
name: dtoParameter.name,
|
||||||
|
type: typeEnum,
|
||||||
|
value: dtoParameter.value ?? "",
|
||||||
|
defaultValue: dtoParameter.defaultValue ?? ""
|
||||||
|
)
|
||||||
|
|
||||||
|
if let replaceIn = dtoParameter.replaceIn {
|
||||||
parameter.replaceIn = replaceIn.components(separatedBy: ",")
|
parameter.replaceIn = replaceIn.components(separatedBy: ",")
|
||||||
}
|
}
|
||||||
|
|
||||||
parameters.append(parameter)
|
return parameter
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
return parameters
|
private func getTagDefinition(
|
||||||
}
|
|
||||||
|
|
||||||
private static func getTagDefinition(
|
|
||||||
id: String,
|
id: String,
|
||||||
name: String,
|
name: String,
|
||||||
type: AnalyticsDefinition.TagType,
|
type: AnalyticsDefinition.TagType,
|
||||||
@@ -64,25 +140,25 @@ class AnalyticsFileParser {
|
|||||||
comments: String?,
|
comments: String?,
|
||||||
parameters: [AnalyticsParameterDTO]?
|
parameters: [AnalyticsParameterDTO]?
|
||||||
) -> AnalyticsDefinition {
|
) -> AnalyticsDefinition {
|
||||||
let definition: AnalyticsDefinition = AnalyticsDefinition(id: id, name: name, type: type)
|
let definition = AnalyticsDefinition(id: id, name: name, type: type)
|
||||||
definition.tags = tags.components(separatedBy: ",")
|
definition.tags = tags
|
||||||
|
.components(separatedBy: ",")
|
||||||
|
.map { $0.removeLeadingTrailingWhitespace() }
|
||||||
|
|
||||||
if let comments = comments {
|
if let comments {
|
||||||
definition.comments = comments
|
definition.comments = comments
|
||||||
}
|
}
|
||||||
|
|
||||||
if let parameters = parameters {
|
if let parameters {
|
||||||
definition.parameters = Self.getParameters(fromData: parameters)
|
definition.parameters = Self.getParameters(from: parameters)
|
||||||
}
|
}
|
||||||
|
|
||||||
return definition
|
return definition
|
||||||
}
|
}
|
||||||
|
|
||||||
private static func getTagDefinitionScreen(fromData screens: [AnalyticsDefinitionScreenDTO]) -> [AnalyticsDefinition] {
|
private func getTagDefinitionScreen(from screens: [AnalyticsDefinitionScreenDTO]) -> [AnalyticsDefinition] {
|
||||||
var definitions: [AnalyticsDefinition] = []
|
screens.map { screen in
|
||||||
|
let definition: AnalyticsDefinition = getTagDefinition(
|
||||||
for screen in screens {
|
|
||||||
let definition: AnalyticsDefinition = Self.getTagDefinition(
|
|
||||||
id: screen.id,
|
id: screen.id,
|
||||||
name: screen.name,
|
name: screen.name,
|
||||||
type: .screen,
|
type: .screen,
|
||||||
@@ -91,28 +167,27 @@ class AnalyticsFileParser {
|
|||||||
parameters: screen.parameters
|
parameters: screen.parameters
|
||||||
)
|
)
|
||||||
|
|
||||||
if target.contains(Analytics.TargetType.matomo.value) {
|
if target.contains(TrackerType.matomo.value) {
|
||||||
// Path
|
// Path
|
||||||
|
|
||||||
guard let path = screen.path else {
|
guard let path = screen.path else {
|
||||||
let error = AnalyticsError.missingElement("screen path")
|
let error = AnalyticsError.missingElement("screen path")
|
||||||
|
print(error.description)
|
||||||
Analytics.exit(withError: error)
|
Analytics.exit(withError: error)
|
||||||
}
|
}
|
||||||
|
|
||||||
|
definition.path = path
|
||||||
|
} else if let path = screen.path {
|
||||||
definition.path = path
|
definition.path = path
|
||||||
}
|
}
|
||||||
|
|
||||||
definitions.append(definition)
|
return definition
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
return definitions
|
private func getTagDefinitionEvent(from events: [AnalyticsDefinitionEventDTO]) -> [AnalyticsDefinition] {
|
||||||
}
|
events.map { event in
|
||||||
|
let definition: AnalyticsDefinition = getTagDefinition(
|
||||||
private static func getTagDefinitionEvent(fromData events: [AnalyticsDefinitionEventDTO]) -> [AnalyticsDefinition] {
|
|
||||||
var definitions: [AnalyticsDefinition] = []
|
|
||||||
|
|
||||||
for event in events {
|
|
||||||
let definition: AnalyticsDefinition = Self.getTagDefinition(
|
|
||||||
id: event.id,
|
id: event.id,
|
||||||
name: event.name,
|
name: event.name,
|
||||||
type: .event,
|
type: .event,
|
||||||
@@ -121,53 +196,35 @@ class AnalyticsFileParser {
|
|||||||
parameters: event.parameters
|
parameters: event.parameters
|
||||||
)
|
)
|
||||||
|
|
||||||
if target.contains(Analytics.TargetType.matomo.value) {
|
if target.contains(TrackerType.matomo.value) {
|
||||||
// Category
|
// Category
|
||||||
|
|
||||||
guard let category = event.category else {
|
guard let category = event.category else {
|
||||||
let error = AnalyticsError.missingElement("event category")
|
let error = AnalyticsError.missingElement("event category")
|
||||||
|
print(error.description)
|
||||||
Analytics.exit(withError: error)
|
Analytics.exit(withError: error)
|
||||||
}
|
}
|
||||||
|
|
||||||
definition.category = category
|
definition.category = category
|
||||||
|
|
||||||
// Action
|
// Action
|
||||||
|
|
||||||
guard let action = event.action else {
|
guard let action = event.action else {
|
||||||
let error = AnalyticsError.missingElement("event action")
|
let error = AnalyticsError.missingElement("event action")
|
||||||
|
print(error.description)
|
||||||
Analytics.exit(withError: error)
|
Analytics.exit(withError: error)
|
||||||
}
|
}
|
||||||
|
|
||||||
definition.action = action
|
definition.action = action
|
||||||
|
} else {
|
||||||
|
if let category = event.category {
|
||||||
|
definition.category = category
|
||||||
}
|
}
|
||||||
|
|
||||||
definitions.append(definition)
|
if let action = event.action {
|
||||||
}
|
definition.action = action
|
||||||
|
}
|
||||||
return definitions
|
}
|
||||||
}
|
|
||||||
|
return definition
|
||||||
static func parse(_ inputFile: String, target: String) -> [AnalyticsCategory] {
|
}
|
||||||
self.inputFile = inputFile
|
|
||||||
self.target = target
|
|
||||||
|
|
||||||
let tagFile: AnalyticsFile = Self.parseYaml()
|
|
||||||
var sections: [AnalyticsCategory] = []
|
|
||||||
|
|
||||||
tagFile.categories.forEach { categorie in
|
|
||||||
let section: AnalyticsCategory = AnalyticsCategory(id: categorie.id)
|
|
||||||
|
|
||||||
if let screens = categorie.screens {
|
|
||||||
section.definitions.append(contentsOf: Self.getTagDefinitionScreen(fromData: screens))
|
|
||||||
}
|
|
||||||
|
|
||||||
if let events = categorie.events {
|
|
||||||
section.definitions.append(contentsOf: Self.getTagDefinitionEvent(fromData: events))
|
|
||||||
}
|
|
||||||
|
|
||||||
sections.append(section)
|
|
||||||
}
|
|
||||||
|
|
||||||
return sections
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@@ -5,15 +5,15 @@
|
|||||||
// Created by Thibaut Schmitt on 20/12/2021.
|
// Created by Thibaut Schmitt on 20/12/2021.
|
||||||
//
|
//
|
||||||
|
|
||||||
import ToolCore
|
@preconcurrency import ArgumentParser
|
||||||
import Foundation
|
import Foundation
|
||||||
import ArgumentParser
|
import ToolCore
|
||||||
|
|
||||||
struct Colors: ParsableCommand {
|
struct Colors: ParsableCommand {
|
||||||
|
|
||||||
// MARK: - CommandConfiguration
|
// MARK: - CommandConfiguration
|
||||||
|
|
||||||
static var configuration = CommandConfiguration(
|
static let configuration = CommandConfiguration(
|
||||||
abstract: "A utility for generate colors assets and their getters.",
|
abstract: "A utility for generate colors assets and their getters.",
|
||||||
version: ResgenSwiftVersion
|
version: ResgenSwiftVersion
|
||||||
)
|
)
|
||||||
@@ -21,8 +21,6 @@ struct Colors: ParsableCommand {
|
|||||||
// MARK: - Static
|
// MARK: - Static
|
||||||
|
|
||||||
static let toolName = "Color"
|
static let toolName = "Color"
|
||||||
static let defaultExtensionName = "UIColor"
|
|
||||||
static let defaultExtensionNameSUI = "Color"
|
|
||||||
static let assetsColorsFolderName = "Colors"
|
static let assetsColorsFolderName = "Colors"
|
||||||
|
|
||||||
// MARK: - Command options
|
// MARK: - Command options
|
||||||
@@ -31,7 +29,7 @@ struct Colors: ParsableCommand {
|
|||||||
|
|
||||||
// MARK: - Run
|
// MARK: - Run
|
||||||
|
|
||||||
public func run() throws {
|
func run() throws {
|
||||||
print("[\(Self.toolName)] Starting colors generation")
|
print("[\(Self.toolName)] Starting colors generation")
|
||||||
|
|
||||||
// Check requirements
|
// Check requirements
|
||||||
@@ -43,28 +41,46 @@ struct Colors: ParsableCommand {
|
|||||||
deleteCurrentColors()
|
deleteCurrentColors()
|
||||||
|
|
||||||
// Get colors to generate
|
// Get colors to generate
|
||||||
let parsedColors = ColorFileParser.parse(options.inputFile,
|
let parsedColors = ColorFileParser.parse(
|
||||||
colorStyle: options.style)
|
options.inputFile,
|
||||||
|
colorStyle: options.style
|
||||||
|
)
|
||||||
// -> Time: 0.0020350217819213867 seconds
|
// -> Time: 0.0020350217819213867 seconds
|
||||||
|
|
||||||
// Generate all colors in xcassets
|
// Generate all colors in xcassets
|
||||||
ColorXcassetHelper.generateXcassetColors(colors: parsedColors,
|
ColorXcassetHelper.generateXcassetColors(
|
||||||
to: options.xcassetsPath)
|
colors: parsedColors,
|
||||||
|
to: options.xcassetsPath
|
||||||
|
)
|
||||||
// -> Time: 3.4505380392074585 seconds
|
// -> Time: 3.4505380392074585 seconds
|
||||||
|
|
||||||
// Generate extension
|
// Generate extension
|
||||||
ColorExtensionGenerator.writeExtensionFile(colors: parsedColors,
|
if let extensionName = options.extensionName,
|
||||||
staticVar: options.staticMembers,
|
let extensionFilePath = options.extensionFilePath {
|
||||||
extensionName: options.extensionName,
|
ColorExtensionGenerator.writeExtensionFile(
|
||||||
extensionFilePath: options.extensionFilePath,
|
colors: parsedColors,
|
||||||
isSwiftUI: false)
|
isStatic: options.staticMembers,
|
||||||
|
extensionName: extensionName,
|
||||||
|
extensionFilePath: extensionFilePath,
|
||||||
|
isSwiftUI: true,
|
||||||
|
visibility: options.extensionVisibility,
|
||||||
|
assetBundle: options.assetBundle
|
||||||
|
)
|
||||||
|
}
|
||||||
|
|
||||||
// Generate extension
|
// Generate extension
|
||||||
ColorExtensionGenerator.writeExtensionFile(colors: parsedColors,
|
if let extensionNameUIKit = options.extensionNameUIKit,
|
||||||
staticVar: options.staticMembers,
|
let extensionFilePathUIKit = options.extensionFilePathUIKit {
|
||||||
extensionName: options.extensionNameSwiftUI,
|
ColorExtensionGenerator.writeExtensionFile(
|
||||||
extensionFilePath: options.extensionFilePathSwiftUI,
|
colors: parsedColors,
|
||||||
isSwiftUI: true)
|
isStatic: options.staticMembers,
|
||||||
|
extensionName: extensionNameUIKit,
|
||||||
|
extensionFilePath: extensionFilePathUIKit,
|
||||||
|
isSwiftUI: false,
|
||||||
|
visibility: options.extensionVisibility,
|
||||||
|
assetBundle: options.assetBundle
|
||||||
|
)
|
||||||
|
}
|
||||||
|
|
||||||
print("[\(Self.toolName)] Colors generated")
|
print("[\(Self.toolName)] Colors generated")
|
||||||
}
|
}
|
||||||
@@ -78,27 +94,51 @@ struct Colors: ParsableCommand {
|
|||||||
guard fileManager.fileExists(atPath: options.inputFile) else {
|
guard fileManager.fileExists(atPath: options.inputFile) else {
|
||||||
let error = ColorsToolError.fileNotExists(options.inputFile)
|
let error = ColorsToolError.fileNotExists(options.inputFile)
|
||||||
print(error.description)
|
print(error.description)
|
||||||
Colors.exit(withError: error)
|
Self.exit(withError: error)
|
||||||
}
|
}
|
||||||
|
|
||||||
// Check if xcassets file exists
|
// Check if xcassets file exists
|
||||||
guard fileManager.fileExists(atPath: options.xcassetsPath) else {
|
guard fileManager.fileExists(atPath: options.xcassetsPath) else {
|
||||||
let error = ColorsToolError.fileNotExists(options.xcassetsPath)
|
let error = ColorsToolError.fileNotExists(options.xcassetsPath)
|
||||||
print(error.description)
|
print(error.description)
|
||||||
Colors.exit(withError: error)
|
Self.exit(withError: error)
|
||||||
}
|
}
|
||||||
|
|
||||||
// Extension for UIKit and SwiftUI should have different name
|
// Extension for UIKit and SwiftUI should have different name (if both are defined)
|
||||||
guard options.extensionName != options.extensionNameSwiftUI else {
|
if let extensionName = options.extensionName,
|
||||||
let error = ColorsToolError.extensionNamesCollision(options.extensionName)
|
let extensionNameUIKit = options.extensionNameUIKit {
|
||||||
|
guard extensionName != extensionNameUIKit else {
|
||||||
|
let error = ColorsToolError.extensionNamesCollision(extensionName)
|
||||||
print(error.description)
|
print(error.description)
|
||||||
Colors.exit(withError: error)
|
Self.exit(withError: error)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// If an extension need to be generated, ensure extensionOutputPath is defined
|
||||||
|
if options.extensionName != nil ||
|
||||||
|
options.extensionNameUIKit != nil {
|
||||||
|
guard let extensionOutputPath = options.extensionOutputPath,
|
||||||
|
extensionOutputPath.isEmpty == false else {
|
||||||
|
let error = ColorsToolError.missingExtensionPath
|
||||||
|
print(error.description)
|
||||||
|
Self.exit(withError: error)
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
// Check if needed to regenerate
|
// Check if needed to regenerate
|
||||||
guard GeneratorChecker.shouldGenerate(force: options.forceGeneration,
|
let fileToCompareToInput: String = {
|
||||||
|
// If there is no extension file to compare
|
||||||
|
// Then check the xcassets file instead
|
||||||
|
if let extensionFilePath = options.extensionFilePath {
|
||||||
|
return extensionFilePath
|
||||||
|
}
|
||||||
|
return options.xcassetsPath
|
||||||
|
}()
|
||||||
|
guard GeneratorChecker.shouldGenerate(
|
||||||
|
force: options.forceGeneration,
|
||||||
inputFilePath: options.inputFile,
|
inputFilePath: options.inputFile,
|
||||||
extensionFilePath: options.extensionFilePath) else {
|
extensionFilePath: fileToCompareToInput
|
||||||
|
) else {
|
||||||
print("[\(Self.toolName)] Colors are already up to date :) ")
|
print("[\(Self.toolName)] Colors are already up to date :) ")
|
||||||
return false
|
return false
|
||||||
}
|
}
|
||||||
@@ -118,7 +158,7 @@ struct Colors: ParsableCommand {
|
|||||||
} catch {
|
} catch {
|
||||||
let error = ColorsToolError.deleteExistingColors("\(options.xcassetsPath)/Colors")
|
let error = ColorsToolError.deleteExistingColors("\(options.xcassetsPath)/Colors")
|
||||||
print(error.description)
|
print(error.description)
|
||||||
Colors.exit(withError: error)
|
Self.exit(withError: error)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@@ -8,6 +8,7 @@
|
|||||||
import Foundation
|
import Foundation
|
||||||
|
|
||||||
enum ColorsToolError: Error {
|
enum ColorsToolError: Error {
|
||||||
|
|
||||||
case extensionNamesCollision(String)
|
case extensionNamesCollision(String)
|
||||||
case badFormat(String)
|
case badFormat(String)
|
||||||
case writeAsset(String)
|
case writeAsset(String)
|
||||||
@@ -16,11 +17,12 @@ enum ColorsToolError: Error {
|
|||||||
case fileNotExists(String)
|
case fileNotExists(String)
|
||||||
case badColorDefinition(String, String)
|
case badColorDefinition(String, String)
|
||||||
case deleteExistingColors(String)
|
case deleteExistingColors(String)
|
||||||
|
case missingExtensionPath
|
||||||
|
|
||||||
var description: String {
|
var description: String {
|
||||||
switch self {
|
switch self {
|
||||||
case .extensionNamesCollision(let extensionName):
|
case .extensionNamesCollision(let extensionName):
|
||||||
return "error: [\(Fonts.toolName)] Error on extension names, extension name and SwiftUI extension name should be different (\(extensionName) is used on both)"
|
return "error: [\(Colors.toolName)] Error on extension names, extension name and SwiftUI extension name should be different (\(extensionName) is used on both)"
|
||||||
|
|
||||||
case .badFormat(let info):
|
case .badFormat(let info):
|
||||||
return "error: [\(Colors.toolName)] Bad line format: \(info). Accepted format are: colorName=\"#RGB/#ARGB\"; colorName \"#RGB/#ARGB\"; colorName \"#RGB/#ARGB\" \"#RGB/#ARGB\""
|
return "error: [\(Colors.toolName)] Bad line format: \(info). Accepted format are: colorName=\"#RGB/#ARGB\"; colorName \"#RGB/#ARGB\"; colorName \"#RGB/#ARGB\" \"#RGB/#ARGB\""
|
||||||
@@ -31,17 +33,20 @@ enum ColorsToolError: Error {
|
|||||||
case .createAssetFolder(let assetsFolder):
|
case .createAssetFolder(let assetsFolder):
|
||||||
return "error: [\(Colors.toolName)] An error occured while creating colors folder `\(assetsFolder)`"
|
return "error: [\(Colors.toolName)] An error occured while creating colors folder `\(assetsFolder)`"
|
||||||
|
|
||||||
case .writeExtension(let filename, let info):
|
case let .writeExtension(filename, info):
|
||||||
return "error: [\(Colors.toolName)] An error occured while writing extension in \(filename): \(info)"
|
return "error: [\(Colors.toolName)] An error occured while writing extension in \(filename): \(info)"
|
||||||
|
|
||||||
case .fileNotExists(let filename):
|
case .fileNotExists(let filename):
|
||||||
return "error: [\(Colors.toolName)] File \(filename) does not exists"
|
return "error: [\(Colors.toolName)] File \(filename) does not exists"
|
||||||
|
|
||||||
case .badColorDefinition(let lightColor, let darkColor):
|
case let .badColorDefinition(lightColor, darkColor):
|
||||||
return "error: [\(Colors.toolName)] One of these two colors has invalid synthax: -\(lightColor)- or -\(darkColor)-"
|
return "error: [\(Colors.toolName)] One of these two colors has invalid synthax: -\(lightColor)- or -\(darkColor)-"
|
||||||
|
|
||||||
case .deleteExistingColors(let assetsFolder):
|
case .deleteExistingColors(let assetsFolder):
|
||||||
return "error: [\(Colors.toolName)] An error occured while deleting colors folder `\(assetsFolder)`"
|
return "error: [\(Colors.toolName)] An error occured while deleting colors folder `\(assetsFolder)`"
|
||||||
|
|
||||||
|
case .missingExtensionPath:
|
||||||
|
return "error: [\(Colors.toolName)] Extension need to be generated but no `extensionOutputPath` is provided"
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@@ -5,33 +5,63 @@
|
|||||||
// Created by Thibaut Schmitt on 17/01/2022.
|
// Created by Thibaut Schmitt on 17/01/2022.
|
||||||
//
|
//
|
||||||
|
|
||||||
import Foundation
|
// CPD-OFF
|
||||||
|
|
||||||
import ArgumentParser
|
import ArgumentParser
|
||||||
|
import Foundation
|
||||||
|
import ToolCore
|
||||||
|
|
||||||
|
// swiftlint:disable no_grouping_extension
|
||||||
|
|
||||||
struct ColorsToolOptions: ParsableArguments {
|
struct ColorsToolOptions: ParsableArguments {
|
||||||
@Flag(name: [.customShort("f"), .customShort("F")], help: "Should force generation")
|
|
||||||
|
@Flag(
|
||||||
|
name: [.customShort("f"), .customShort("F")],
|
||||||
|
help: "Should force generation"
|
||||||
|
)
|
||||||
var forceGeneration = false
|
var forceGeneration = false
|
||||||
|
|
||||||
@Argument(help: "Input files where colors ared defined.", transform: { $0.replaceTiltWithHomeDirectoryPath() })
|
@Argument(
|
||||||
|
help: "Input files where colors ared defined.",
|
||||||
|
transform: { $0.replaceTiltWithHomeDirectoryPath() }
|
||||||
|
)
|
||||||
var inputFile: String
|
var inputFile: String
|
||||||
|
|
||||||
@Option(help: "Color style to generate: light for light colors only, or all for dark and light colors")
|
@Option(help: "Color style to generate: light for light colors only, or all for dark and light colors")
|
||||||
var style: ColorStyle
|
var style: ColorStyle
|
||||||
|
|
||||||
@Option(help: "Path of xcassets where to generate colors", transform: { $0.replaceTiltWithHomeDirectoryPath() })
|
@Option(
|
||||||
|
help: "Path of xcassets where to generate colors",
|
||||||
|
transform: { $0.replaceTiltWithHomeDirectoryPath() }
|
||||||
|
)
|
||||||
var xcassetsPath: String
|
var xcassetsPath: String
|
||||||
|
|
||||||
@Option(help: "Path where to generate the extension.", transform: { $0.replaceTiltWithHomeDirectoryPath() })
|
|
||||||
var extensionOutputPath: String
|
|
||||||
|
|
||||||
@Option(help: "Tell if it will generate static properties or not")
|
@Option(help: "Tell if it will generate static properties or not")
|
||||||
var staticMembers: Bool = false
|
var staticMembers: Bool = false
|
||||||
|
|
||||||
@Option(help: "Extension name. If not specified, it will generate an UIColor extension.")
|
@Option(
|
||||||
var extensionName: String = Colors.defaultExtensionName
|
name: .customLong("visibility"),
|
||||||
|
help: "Visibility of extension and properties. Possibles values: public, private, package, internal. Default is internal",
|
||||||
|
completion: .list(["public", "private", "package", "internal"])
|
||||||
|
)
|
||||||
|
var extensionVisibility: ExtensionVisibility = .internal
|
||||||
|
|
||||||
@Option(help: "SwiftUI Extension name. If not specified, it will generate an Color extension.")
|
@Option(
|
||||||
var extensionNameSwiftUI: String = Colors.defaultExtensionNameSUI
|
help: "Bundle where the asset are generated"
|
||||||
|
)
|
||||||
|
var assetBundle: AssetBundle = .main
|
||||||
|
|
||||||
|
@Option(
|
||||||
|
help: "Path where to generate the extension.",
|
||||||
|
transform: { $0.replaceTiltWithHomeDirectoryPath() }
|
||||||
|
)
|
||||||
|
var extensionOutputPath: String?
|
||||||
|
|
||||||
|
@Option(help: "SwiftUI extension name. If not specified, no extension will be generated.")
|
||||||
|
var extensionName: String?
|
||||||
|
|
||||||
|
@Option(help: "UIKit extension name. If not specified, no extension will be generated.")
|
||||||
|
var extensionNameUIKit: String?
|
||||||
|
|
||||||
@Option(help: "Extension suffix. Ex: MyApp, it will generate {extensionName}+ColorsMyApp.swift")
|
@Option(help: "Extension suffix. Ex: MyApp, it will generate {extensionName}+ColorsMyApp.swift")
|
||||||
var extensionSuffix: String?
|
var extensionSuffix: String?
|
||||||
@@ -41,29 +71,37 @@ struct ColorsToolOptions: ParsableArguments {
|
|||||||
|
|
||||||
extension ColorsToolOptions {
|
extension ColorsToolOptions {
|
||||||
|
|
||||||
// MARK: - UIKit
|
// MARK: - SwiftUI
|
||||||
|
|
||||||
var extensionFileName: String {
|
var extensionFileName: String? {
|
||||||
if let extensionSuffix = extensionSuffix {
|
guard let extensionName else { return nil }
|
||||||
|
|
||||||
|
if let extensionSuffix {
|
||||||
return "\(extensionName)+\(extensionSuffix).swift"
|
return "\(extensionName)+\(extensionSuffix).swift"
|
||||||
}
|
}
|
||||||
return "\(extensionName).swift"
|
return "\(extensionName).swift"
|
||||||
}
|
}
|
||||||
|
|
||||||
var extensionFilePath: String {
|
var extensionFilePath: String? {
|
||||||
"\(extensionOutputPath)/\(extensionFileName)"
|
guard let extensionOutputPath, let extensionFileName else { return nil }
|
||||||
|
|
||||||
|
return "\(extensionOutputPath)/\(extensionFileName)"
|
||||||
}
|
}
|
||||||
|
|
||||||
// MARK: - SwiftUI
|
// MARK: - UIKit
|
||||||
|
|
||||||
var extensionFileNameSwiftUI: String {
|
var extensionFileNameUIKit: String? {
|
||||||
if let extensionSuffix = extensionSuffix {
|
guard let extensionNameUIKit else { return nil }
|
||||||
return "\(extensionNameSwiftUI)+\(extensionSuffix).swift"
|
|
||||||
|
if let extensionSuffix {
|
||||||
|
return "\(extensionNameUIKit)+\(extensionSuffix).swift"
|
||||||
}
|
}
|
||||||
return "\(extensionNameSwiftUI).swift"
|
return "\(extensionNameUIKit).swift"
|
||||||
}
|
}
|
||||||
|
|
||||||
var extensionFilePathSwiftUI: String {
|
var extensionFilePathUIKit: String? {
|
||||||
"\(extensionOutputPath)/\(extensionFileNameSwiftUI)"
|
guard let extensionOutputPath, let extensionFileNameUIKit else { return nil }
|
||||||
}
|
|
||||||
|
return "\(extensionOutputPath)/\(extensionFileNameUIKit)"
|
||||||
}
|
}
|
||||||
|
}// CPD-ON
|
||||||
|
@@ -15,35 +15,56 @@ struct ColorExtensionGenerator {
|
|||||||
|
|
||||||
// MARK: - UIKit
|
// MARK: - UIKit
|
||||||
|
|
||||||
static func writeExtensionFile(colors: [ParsedColor],
|
static func writeExtensionFile(
|
||||||
staticVar: Bool,
|
colors: [ParsedColor],
|
||||||
|
isStatic: Bool,
|
||||||
extensionName: String,
|
extensionName: String,
|
||||||
extensionFilePath: String,
|
extensionFilePath: String,
|
||||||
isSwiftUI: Bool) {
|
isSwiftUI: Bool,
|
||||||
|
visibility: ExtensionVisibility,
|
||||||
|
assetBundle: AssetBundle
|
||||||
|
) {
|
||||||
// Create extension content
|
// Create extension content
|
||||||
let extensionContent = Self.getExtensionContent(colors: colors,
|
let extensionContent = Self.getExtensionContent(
|
||||||
staticVar: staticVar,
|
colors: colors,
|
||||||
|
isStatic: isStatic,
|
||||||
extensionName: extensionName,
|
extensionName: extensionName,
|
||||||
isSwiftUI: isSwiftUI)
|
isSwiftUI: isSwiftUI,
|
||||||
|
visibility: visibility,
|
||||||
|
assetBundle: assetBundle
|
||||||
|
)
|
||||||
|
|
||||||
// Write content
|
// Write content
|
||||||
let extensionFilePathURL = URL(fileURLWithPath: extensionFilePath)
|
let extensionFilePathURL = URL(fileURLWithPath: extensionFilePath)
|
||||||
do {
|
do {
|
||||||
try extensionContent.write(to: extensionFilePathURL, atomically: false, encoding: .utf8)
|
try extensionContent.write(to: extensionFilePathURL, atomically: false, encoding: .utf8)
|
||||||
} catch (let error) {
|
} catch {
|
||||||
let error = ColorsToolError.writeExtension(extensionFilePath, error.localizedDescription)
|
let error = ColorsToolError.writeExtension(extensionFilePath, error.localizedDescription)
|
||||||
print(error.description)
|
print(error.description)
|
||||||
Colors.exit(withError: error)
|
Colors.exit(withError: error)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
static func getExtensionContent(colors: [ParsedColor],
|
static func getExtensionContent(
|
||||||
staticVar: Bool,
|
colors: [ParsedColor],
|
||||||
|
isStatic: Bool,
|
||||||
extensionName: String,
|
extensionName: String,
|
||||||
isSwiftUI: Bool) -> String {
|
isSwiftUI: Bool,
|
||||||
|
visibility: ExtensionVisibility,
|
||||||
|
assetBundle: AssetBundle
|
||||||
|
) -> String {
|
||||||
[
|
[
|
||||||
Self.getHeader(extensionClassname: extensionName, isSwiftUI: isSwiftUI),
|
Self.getHeader(
|
||||||
Self.getProperties(for: colors, withStaticVar: staticVar, isSwiftUI: isSwiftUI),
|
extensionClassname: extensionName,
|
||||||
|
isSwiftUI: isSwiftUI
|
||||||
|
),
|
||||||
|
Self.getProperties(
|
||||||
|
for: colors,
|
||||||
|
withIsStatic: isStatic,
|
||||||
|
isSwiftUI: isSwiftUI,
|
||||||
|
visibility: visibility,
|
||||||
|
assetBundle: assetBundle
|
||||||
|
),
|
||||||
Self.getFooter()
|
Self.getFooter()
|
||||||
]
|
]
|
||||||
.joined(separator: "\n")
|
.joined(separator: "\n")
|
||||||
@@ -66,11 +87,20 @@ struct ColorExtensionGenerator {
|
|||||||
"""
|
"""
|
||||||
}
|
}
|
||||||
|
|
||||||
private static func getProperties(for colors: [ParsedColor],
|
private static func getProperties(
|
||||||
withStaticVar staticVar: Bool,
|
for colors: [ParsedColor],
|
||||||
isSwiftUI: Bool) -> String {
|
withIsStatic isStatic: Bool,
|
||||||
|
isSwiftUI: Bool,
|
||||||
|
visibility: ExtensionVisibility,
|
||||||
|
assetBundle: AssetBundle
|
||||||
|
) -> String {
|
||||||
colors.map {
|
colors.map {
|
||||||
$0.getColorProperty(isStatic: staticVar, isSwiftUI: isSwiftUI)
|
$0.getColorProperty(
|
||||||
|
isStatic: isStatic,
|
||||||
|
isSwiftUI: isSwiftUI,
|
||||||
|
visibility: visibility,
|
||||||
|
assetBundle: assetBundle
|
||||||
|
)
|
||||||
}
|
}
|
||||||
.joined(separator: "\n\n")
|
.joined(separator: "\n\n")
|
||||||
}
|
}
|
||||||
|
@@ -8,7 +8,7 @@
|
|||||||
import Foundation
|
import Foundation
|
||||||
import ToolCore
|
import ToolCore
|
||||||
|
|
||||||
struct ColorXcassetHelper {
|
enum ColorXcassetHelper {
|
||||||
|
|
||||||
static func generateXcassetColors(colors: [ParsedColor], to xcassetsPath: String) {
|
static func generateXcassetColors(colors: [ParsedColor], to xcassetsPath: String) {
|
||||||
colors.forEach {
|
colors.forEach {
|
||||||
@@ -25,8 +25,10 @@ struct ColorXcassetHelper {
|
|||||||
let fileManager = FileManager()
|
let fileManager = FileManager()
|
||||||
if fileManager.fileExists(atPath: colorSetPath) == false {
|
if fileManager.fileExists(atPath: colorSetPath) == false {
|
||||||
do {
|
do {
|
||||||
try fileManager.createDirectory(atPath: colorSetPath,
|
try fileManager.createDirectory(
|
||||||
withIntermediateDirectories: true)
|
atPath: colorSetPath,
|
||||||
|
withIntermediateDirectories: true
|
||||||
|
)
|
||||||
} catch {
|
} catch {
|
||||||
let error = ColorsToolError.createAssetFolder(colorSetPath)
|
let error = ColorsToolError.createAssetFolder(colorSetPath)
|
||||||
print(error.description)
|
print(error.description)
|
||||||
@@ -38,7 +40,7 @@ struct ColorXcassetHelper {
|
|||||||
let contentsJsonPathURL = URL(fileURLWithPath: contentsJsonPath)
|
let contentsJsonPathURL = URL(fileURLWithPath: contentsJsonPath)
|
||||||
do {
|
do {
|
||||||
try color.contentsJSON().write(to: contentsJsonPathURL, atomically: false, encoding: .utf8)
|
try color.contentsJSON().write(to: contentsJsonPathURL, atomically: false, encoding: .utf8)
|
||||||
} catch (let error) {
|
} catch {
|
||||||
let error = ColorsToolError.writeAsset(error.localizedDescription)
|
let error = ColorsToolError.writeAsset(error.localizedDescription)
|
||||||
print(error.description)
|
print(error.description)
|
||||||
Colors.exit(withError: error)
|
Colors.exit(withError: error)
|
||||||
|
@@ -5,10 +5,11 @@
|
|||||||
// Created by Thibaut Schmitt on 29/08/2022.
|
// Created by Thibaut Schmitt on 29/08/2022.
|
||||||
//
|
//
|
||||||
|
|
||||||
import Foundation
|
|
||||||
import ArgumentParser
|
import ArgumentParser
|
||||||
|
import Foundation
|
||||||
|
|
||||||
enum ColorStyle: String, Decodable, ExpressibleByArgument {
|
enum ColorStyle: String, Decodable, ExpressibleByArgument {
|
||||||
|
|
||||||
case light
|
case light
|
||||||
case all
|
case all
|
||||||
|
|
||||||
|
@@ -6,13 +6,19 @@
|
|||||||
//
|
//
|
||||||
|
|
||||||
import Foundation
|
import Foundation
|
||||||
|
import ToolCore
|
||||||
|
|
||||||
struct ParsedColor {
|
struct ParsedColor {
|
||||||
|
|
||||||
|
// MARK: - Properties
|
||||||
|
|
||||||
let name: String
|
let name: String
|
||||||
let light: String
|
let light: String
|
||||||
let dark: String
|
let dark: String
|
||||||
|
|
||||||
// Generate Contents.json content
|
// MARK: - Contents.json
|
||||||
|
|
||||||
|
/// Generate Contents.json content
|
||||||
func contentsJSON() -> String {
|
func contentsJSON() -> String {
|
||||||
let lightARGB = light.colorComponent()
|
let lightARGB = light.colorComponent()
|
||||||
let darkARGB = dark.colorComponent()
|
let darkARGB = dark.colorComponent()
|
||||||
@@ -72,21 +78,26 @@ struct ParsedColor {
|
|||||||
"""
|
"""
|
||||||
}
|
}
|
||||||
|
|
||||||
// MARK: - UIKit
|
// MARK: - Property
|
||||||
|
|
||||||
func getColorProperty(isStatic: Bool, isSwiftUI: Bool) -> String {
|
func getColorProperty(
|
||||||
|
isStatic: Bool,
|
||||||
|
isSwiftUI: Bool,
|
||||||
|
visibility: ExtensionVisibility,
|
||||||
|
assetBundle: AssetBundle
|
||||||
|
) -> String {
|
||||||
if isSwiftUI {
|
if isSwiftUI {
|
||||||
return """
|
return """
|
||||||
/// Color \(name) is \(light) (light) or \(dark) (dark)"
|
/// Color \(name) is \(light) (light) or \(dark) (dark)"
|
||||||
\(isStatic ? "static " : "")var \(name): Color {
|
\(visibility) \(isStatic ? "static " : "")var \(name): Color {
|
||||||
Color("\(name)")
|
Color("\(name)", bundle: Bundle.\(assetBundle))
|
||||||
}
|
}
|
||||||
"""
|
"""
|
||||||
}
|
}
|
||||||
return """
|
return """
|
||||||
/// Color \(name) is \(light) (light) or \(dark) (dark)"
|
/// Color \(name) is \(light) (light) or \(dark) (dark)"
|
||||||
\(isStatic ? "static " : "@objc ")var \(name): UIColor {
|
\(isStatic ? "" : "@objc ")\(visibility) \(isStatic ? "static " : "")var \(name): UIColor {
|
||||||
UIColor(named: "\(name)")!
|
UIColor(named: "\(name)", in: Bundle.\(assetBundle), compatibleWith: nil)!
|
||||||
}
|
}
|
||||||
"""
|
"""
|
||||||
}
|
}
|
||||||
|
@@ -7,10 +7,11 @@
|
|||||||
|
|
||||||
import Foundation
|
import Foundation
|
||||||
|
|
||||||
class ColorFileParser {
|
enum ColorFileParser {
|
||||||
|
|
||||||
static func parse(_ inputFile: String, colorStyle: ColorStyle) -> [ParsedColor] {
|
static func parse(_ inputFile: String, colorStyle: ColorStyle) -> [ParsedColor] {
|
||||||
// Get content of input file
|
// Get content of input file
|
||||||
let inputFileContent = try! String(contentsOfFile: inputFile, encoding: .utf8)
|
let inputFileContent = try! String(contentsOfFile: inputFile, encoding: .utf8) // swiftlint:disable:this force_try
|
||||||
let colorsByLines = inputFileContent.components(separatedBy: CharacterSet.newlines)
|
let colorsByLines = inputFileContent.components(separatedBy: CharacterSet.newlines)
|
||||||
|
|
||||||
// Iterate on each line of input file
|
// Iterate on each line of input file
|
||||||
@@ -20,7 +21,7 @@ class ColorFileParser {
|
|||||||
static func parseLines(lines: [String], colorStyle: ColorStyle) -> [ParsedColor] {
|
static func parseLines(lines: [String], colorStyle: ColorStyle) -> [ParsedColor] {
|
||||||
lines
|
lines
|
||||||
.enumerated()
|
.enumerated()
|
||||||
.compactMap { lineNumber, colorLine in
|
.compactMap { _, colorLine in // swiftlint:disable:this unused_enumerated
|
||||||
// Required format:
|
// Required format:
|
||||||
// colorName = "#RGB/#ARGB", colorName "#RGB/#ARGB", colorName "#RGB/#ARGB" "#RGB/#ARGB"
|
// colorName = "#RGB/#ARGB", colorName "#RGB/#ARGB", colorName "#RGB/#ARGB" "#RGB/#ARGB"
|
||||||
let colorLineCleanedUp = colorLine
|
let colorLineCleanedUp = colorLine
|
||||||
|
@@ -5,30 +5,41 @@
|
|||||||
// Created by Thibaut Schmitt on 17/01/2022.
|
// Created by Thibaut Schmitt on 17/01/2022.
|
||||||
//
|
//
|
||||||
|
|
||||||
import Foundation
|
|
||||||
import ArgumentParser
|
import ArgumentParser
|
||||||
|
import Foundation
|
||||||
|
import ToolCore
|
||||||
|
|
||||||
|
// swiftlint:disable no_grouping_extension
|
||||||
|
|
||||||
struct FontsOptions: ParsableArguments {
|
struct FontsOptions: ParsableArguments {
|
||||||
|
|
||||||
@Flag(name: [.customShort("f"), .customShort("F")], help: "Should force generation")
|
@Flag(name: [.customShort("f"), .customShort("F")], help: "Should force generation")
|
||||||
var forceGeneration = false
|
var forceGeneration = false
|
||||||
|
|
||||||
@Argument(help: "Input files where fonts ared defined.", transform: { $0.replaceTiltWithHomeDirectoryPath() })
|
@Argument(help: "Input files where fonts ared defined.", transform: { $0.replaceTiltWithHomeDirectoryPath() })
|
||||||
var inputFile: String
|
var inputFile: String
|
||||||
|
|
||||||
@Option(help: "Path where to generate the extension.", transform: { $0.replaceTiltWithHomeDirectoryPath() })
|
|
||||||
var extensionOutputPath: String
|
|
||||||
|
|
||||||
@Option(help: "Tell if it will generate static properties or methods")
|
@Option(help: "Tell if it will generate static properties or methods")
|
||||||
var staticMembers: Bool = false
|
var staticMembers: Bool = false
|
||||||
|
|
||||||
@Option(help: "Extension name. If not specified, it will generate an UIFont extension.")
|
@Option(
|
||||||
var extensionName: String = Fonts.defaultExtensionName
|
name: .customLong("visibility"),
|
||||||
|
help: "Visibility of extension and properties. Possibles values: public, private, package, internal. Default is internal",
|
||||||
|
completion: .list(["public", "private", "package", "internal"])
|
||||||
|
)
|
||||||
|
var extensionVisibility: ExtensionVisibility = .internal
|
||||||
|
|
||||||
|
@Option(help: "Path where to generate the extension.", transform: { $0.replaceTiltWithHomeDirectoryPath() })
|
||||||
|
var extensionOutputPath: String
|
||||||
|
|
||||||
@Option(help: "Extension name. If not specified, it will generate an Font extension.")
|
@Option(help: "Extension name. If not specified, it will generate an Font extension.")
|
||||||
var extensionNameSwiftUI: String = Fonts.defaultExtensionNameSUI
|
var extensionName: String = Fonts.defaultExtensionName
|
||||||
|
|
||||||
|
@Option(help: "Extension name. If not specified, no extension will be generated.")
|
||||||
|
var extensionNameUIKit: String?
|
||||||
|
|
||||||
@Option(help: "Extension suffix. Ex: MyApp, it will generate {extensionName}+FontsMyApp.swift")
|
@Option(help: "Extension suffix. Ex: MyApp, it will generate {extensionName}+FontsMyApp.swift")
|
||||||
var extensionSuffix: String = ""
|
var extensionSuffix: String?
|
||||||
|
|
||||||
@Option(name: .customLong("info-plist-paths"), help: "Info.plist paths (array). Will be used to update UIAppFonts content")
|
@Option(name: .customLong("info-plist-paths"), help: "Info.plist paths (array). Will be used to update UIAppFonts content")
|
||||||
fileprivate var infoPlistPathsRaw: String = ""
|
fileprivate var infoPlistPathsRaw: String = ""
|
||||||
@@ -38,10 +49,10 @@ struct FontsOptions: ParsableArguments {
|
|||||||
|
|
||||||
extension FontsOptions {
|
extension FontsOptions {
|
||||||
|
|
||||||
// MARK: - UIKit
|
// MARK: - SwiftUI
|
||||||
|
|
||||||
var extensionFileName: String {
|
var extensionFileName: String {
|
||||||
if extensionSuffix.isEmpty == false {
|
if let extensionSuffix {
|
||||||
return "\(extensionName)+\(extensionSuffix).swift"
|
return "\(extensionName)+\(extensionSuffix).swift"
|
||||||
}
|
}
|
||||||
return "\(extensionName).swift"
|
return "\(extensionName).swift"
|
||||||
@@ -51,17 +62,21 @@ extension FontsOptions {
|
|||||||
"\(extensionOutputPath)/\(extensionFileName)"
|
"\(extensionOutputPath)/\(extensionFileName)"
|
||||||
}
|
}
|
||||||
|
|
||||||
// MARK: - SwiftUI
|
// MARK: - UIKit
|
||||||
|
|
||||||
var extensionFileNameSwiftUI: String {
|
var extensionFileNameUIKit: String? {
|
||||||
if extensionSuffix.isEmpty == false {
|
guard let extensionNameUIKit else { return nil }
|
||||||
return "\(extensionNameSwiftUI)+\(extensionSuffix).swift"
|
|
||||||
|
if let extensionSuffix {
|
||||||
|
return "\(extensionNameUIKit)+\(extensionSuffix).swift"
|
||||||
}
|
}
|
||||||
return "\(extensionNameSwiftUI).swift"
|
return "\(extensionNameUIKit).swift"
|
||||||
}
|
}
|
||||||
|
|
||||||
var extensionFilePathSwiftUI: String {
|
var extensionFilePathUIKit: String? {
|
||||||
"\(extensionOutputPath)/\(extensionFileNameSwiftUI)"
|
guard let extensionFileNameUIKit else { return nil }
|
||||||
|
|
||||||
|
return "\(extensionOutputPath)/\(extensionFileNameUIKit)"
|
||||||
}
|
}
|
||||||
|
|
||||||
// MARK: -
|
// MARK: -
|
||||||
|
@@ -5,15 +5,15 @@
|
|||||||
// Created by Thibaut Schmitt on 13/12/2021.
|
// Created by Thibaut Schmitt on 13/12/2021.
|
||||||
//
|
//
|
||||||
|
|
||||||
import ToolCore
|
|
||||||
import Foundation
|
|
||||||
import ArgumentParser
|
import ArgumentParser
|
||||||
|
import Foundation
|
||||||
|
import ToolCore
|
||||||
|
|
||||||
struct Fonts: ParsableCommand {
|
struct Fonts: ParsableCommand {
|
||||||
|
|
||||||
// MARK: - CommandConfiguration
|
// MARK: - CommandConfiguration
|
||||||
|
|
||||||
static var configuration = CommandConfiguration(
|
static let configuration = CommandConfiguration(
|
||||||
abstract: "A utility to generate an helpful etension to access your custom font from code and also Info.plist UIAppsFont content.",
|
abstract: "A utility to generate an helpful etension to access your custom font from code and also Info.plist UIAppsFont content.",
|
||||||
version: ResgenSwiftVersion
|
version: ResgenSwiftVersion
|
||||||
)
|
)
|
||||||
@@ -21,8 +21,7 @@ struct Fonts: ParsableCommand {
|
|||||||
// MARK: - Static
|
// MARK: - Static
|
||||||
|
|
||||||
static let toolName = "Fonts"
|
static let toolName = "Fonts"
|
||||||
static let defaultExtensionName = "UIFont"
|
static let defaultExtensionName = "Font"
|
||||||
static let defaultExtensionNameSUI = "Font"
|
|
||||||
|
|
||||||
// MARK: - Command Options
|
// MARK: - Command Options
|
||||||
|
|
||||||
@@ -30,7 +29,7 @@ struct Fonts: ParsableCommand {
|
|||||||
|
|
||||||
// MARK: - Run
|
// MARK: - Run
|
||||||
|
|
||||||
public func run() throws {
|
func run() throws {
|
||||||
print("[\(Self.toolName)] Starting fonts generation")
|
print("[\(Self.toolName)] Starting fonts generation")
|
||||||
print("[\(Self.toolName)] Will use inputFile \(options.inputFile) to generate fonts")
|
print("[\(Self.toolName)] Will use inputFile \(options.inputFile) to generate fonts")
|
||||||
|
|
||||||
@@ -43,25 +42,45 @@ struct Fonts: ParsableCommand {
|
|||||||
let fontsToGenerate = FontFileParser.parse(options.inputFile)
|
let fontsToGenerate = FontFileParser.parse(options.inputFile)
|
||||||
|
|
||||||
// Get real font names
|
// Get real font names
|
||||||
let inputFolder = URL(fileURLWithPath: options.inputFile).deletingLastPathComponent().relativePath
|
let inputFolder = URL(fileURLWithPath: options.inputFile)
|
||||||
let fontsNames = FontsToolHelper.getFontPostScriptName(for: fontsToGenerate,
|
.deletingLastPathComponent()
|
||||||
inputFolder: inputFolder)
|
.relativePath
|
||||||
|
|
||||||
|
let fontsNames = FontsToolHelper.getFontPostScriptName(
|
||||||
|
for: fontsToGenerate,
|
||||||
|
inputFolder: inputFolder
|
||||||
|
)
|
||||||
|
|
||||||
// Generate extension
|
// Generate extension
|
||||||
FontExtensionGenerator.writeExtensionFile(fontsNames: fontsNames,
|
FontExtensionGenerator.writeExtensionFile(
|
||||||
staticVar: options.staticMembers,
|
fontsNames: fontsNames,
|
||||||
|
isStatic: options.staticMembers,
|
||||||
extensionName: options.extensionName,
|
extensionName: options.extensionName,
|
||||||
extensionFilePath: options.extensionFilePath,
|
extensionFilePath: options.extensionFilePath,
|
||||||
isSwiftUI: false)
|
isSwiftUI: true,
|
||||||
|
visibility: options.extensionVisibility
|
||||||
|
)
|
||||||
|
|
||||||
FontExtensionGenerator.writeExtensionFile(fontsNames: fontsNames,
|
if let extensionNameUIKit = options.extensionNameUIKit,
|
||||||
staticVar: options.staticMembers,
|
let extensionFilePathUIKit = options.extensionFilePathUIKit {
|
||||||
extensionName: options.extensionNameSwiftUI,
|
FontExtensionGenerator.writeExtensionFile(
|
||||||
extensionFilePath: options.extensionFilePathSwiftUI,
|
fontsNames: fontsNames,
|
||||||
isSwiftUI: true)
|
isStatic: options.staticMembers,
|
||||||
|
extensionName: extensionNameUIKit,
|
||||||
|
extensionFilePath: extensionFilePathUIKit,
|
||||||
|
isSwiftUI: false,
|
||||||
|
visibility: options.extensionVisibility
|
||||||
|
)
|
||||||
|
}
|
||||||
|
|
||||||
|
if options.infoPlistPaths.isEmpty == false {
|
||||||
|
let plistUpdateFontsData = FontPlistGenerator.generatePlistUIAppsFontContent(
|
||||||
|
for: fontsNames,
|
||||||
|
infoPlistPaths: options.infoPlistPaths
|
||||||
|
)
|
||||||
print("Info.plist has been updated with:")
|
print("Info.plist has been updated with:")
|
||||||
print("\(FontPlistGenerator.generatePlistUIAppsFontContent(for: fontsNames, infoPlistPaths: options.infoPlistPaths))")
|
print(plistUpdateFontsData)
|
||||||
|
}
|
||||||
|
|
||||||
print("[\(Self.toolName)] Fonts generated")
|
print("[\(Self.toolName)] Fonts generated")
|
||||||
}
|
}
|
||||||
@@ -75,20 +94,22 @@ struct Fonts: ParsableCommand {
|
|||||||
guard fileManager.fileExists(atPath: options.inputFile) else {
|
guard fileManager.fileExists(atPath: options.inputFile) else {
|
||||||
let error = FontsToolError.fileNotExists(options.inputFile)
|
let error = FontsToolError.fileNotExists(options.inputFile)
|
||||||
print(error.description)
|
print(error.description)
|
||||||
Fonts.exit(withError: error)
|
Self.exit(withError: error)
|
||||||
}
|
}
|
||||||
|
|
||||||
// Extension for UIKit and SwiftUI should have different name
|
// Extension for UIKit and SwiftUI should have different name
|
||||||
guard options.extensionName != options.extensionNameSwiftUI else {
|
guard options.extensionName != options.extensionNameUIKit else {
|
||||||
let error = FontsToolError.extensionNamesCollision(options.extensionName)
|
let error = FontsToolError.extensionNamesCollision(options.extensionName)
|
||||||
print(error.description)
|
print(error.description)
|
||||||
Fonts.exit(withError: error)
|
Self.exit(withError: error)
|
||||||
}
|
}
|
||||||
|
|
||||||
// Check if needed to regenerate
|
// Check if needed to regenerate
|
||||||
guard GeneratorChecker.shouldGenerate(force: options.forceGeneration,
|
guard GeneratorChecker.shouldGenerate(
|
||||||
|
force: options.forceGeneration,
|
||||||
inputFilePath: options.inputFile,
|
inputFilePath: options.inputFile,
|
||||||
extensionFilePath: options.extensionFilePath) else {
|
extensionFilePath: options.extensionFilePath
|
||||||
|
) else {
|
||||||
print("[\(Self.toolName)] Fonts are already up to date :) ")
|
print("[\(Self.toolName)] Fonts are already up to date :) ")
|
||||||
return false
|
return false
|
||||||
}
|
}
|
||||||
|
@@ -8,18 +8,20 @@
|
|||||||
import Foundation
|
import Foundation
|
||||||
|
|
||||||
enum FontsToolError: Error {
|
enum FontsToolError: Error {
|
||||||
|
|
||||||
case extensionNamesCollision(String)
|
case extensionNamesCollision(String)
|
||||||
case fcScan(String, Int32, String?)
|
case fcScan(String, Int32, String?)
|
||||||
case inputFolderNotFound(String)
|
case inputFolderNotFound(String)
|
||||||
case fileNotExists(String)
|
case fileNotExists(String)
|
||||||
case writeExtension(String, String)
|
case writeExtension(String, String)
|
||||||
|
case missingExtensionPath
|
||||||
|
|
||||||
var description: String {
|
var description: String {
|
||||||
switch self {
|
switch self {
|
||||||
case .extensionNamesCollision(let extensionName):
|
case .extensionNamesCollision(let extensionName):
|
||||||
return "error: [\(Fonts.toolName)] Error on extension names, extension name and SwiftUI extension name should be different (\(extensionName) is used on both)"
|
return "error: [\(Fonts.toolName)] Error on extension names, extension name and SwiftUI extension name should be different (\(extensionName) is used on both)"
|
||||||
|
|
||||||
case .fcScan(let path, let code, let output):
|
case let .fcScan(path, code, output):
|
||||||
return "error: [\(Fonts.toolName)] Error while getting fontName (fc-scan --format %{postscriptname} \(path). fc-scan exit with \(code) and output is: \(output ?? "no output")"
|
return "error: [\(Fonts.toolName)] Error while getting fontName (fc-scan --format %{postscriptname} \(path). fc-scan exit with \(code) and output is: \(output ?? "no output")"
|
||||||
|
|
||||||
case .inputFolderNotFound(let inputFolder):
|
case .inputFolderNotFound(let inputFolder):
|
||||||
@@ -28,8 +30,11 @@ enum FontsToolError: Error {
|
|||||||
case .fileNotExists(let filename):
|
case .fileNotExists(let filename):
|
||||||
return "error: [\(Fonts.toolName)] File \(filename) does not exists"
|
return "error: [\(Fonts.toolName)] File \(filename) does not exists"
|
||||||
|
|
||||||
case .writeExtension(let filename, let info):
|
case let .writeExtension(filename, info):
|
||||||
return "error: [\(Fonts.toolName)] An error occured while writing extension in \(filename): \(info)"
|
return "error: [\(Fonts.toolName)] An error occured while writing extension in \(filename): \(info)"
|
||||||
|
|
||||||
|
case .missingExtensionPath:
|
||||||
|
return "error: [\(Fonts.toolName)] Extension need to be generated but no `extensionOutputPath` is provided"
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@@ -8,7 +8,7 @@
|
|||||||
import Foundation
|
import Foundation
|
||||||
import ToolCore
|
import ToolCore
|
||||||
|
|
||||||
class FontsToolHelper {
|
enum FontsToolHelper {
|
||||||
|
|
||||||
static func getFontPostScriptName(for fonts: [String], inputFolder: String) -> [FontName] {
|
static func getFontPostScriptName(for fonts: [String], inputFolder: String) -> [FontName] {
|
||||||
let fontsFilenames = Self.getFontsFilenames(fromInputFolder: inputFolder)
|
let fontsFilenames = Self.getFontsFilenames(fromInputFolder: inputFolder)
|
||||||
@@ -43,10 +43,10 @@ class FontsToolHelper {
|
|||||||
Fonts.exit(withError: error)
|
Fonts.exit(withError: error)
|
||||||
}
|
}
|
||||||
|
|
||||||
let enumerator: FileManager.DirectoryEnumerator = fileManager.enumerator(atPath: inputFolder)!
|
let enumerator: FileManager.DirectoryEnumerator = fileManager.enumerator(atPath: inputFolder)! // swiftlint:disable:this force_unwrapping
|
||||||
|
|
||||||
// Filters font files
|
// Filters font files
|
||||||
let fontsFileNames: [String] = (enumerator.allObjects as! [String])
|
let fontsFileNames: [String] = (enumerator.allObjects as! [String]) // swiftlint:disable:this force_cast
|
||||||
.filter {
|
.filter {
|
||||||
if $0.hasSuffix(".ttf") || $0.hasSuffix(".otf") {
|
if $0.hasSuffix(".ttf") || $0.hasSuffix(".otf") {
|
||||||
return true
|
return true
|
||||||
@@ -57,17 +57,26 @@ class FontsToolHelper {
|
|||||||
return fontsFileNames
|
return fontsFileNames
|
||||||
}
|
}
|
||||||
|
|
||||||
private static func getFontName(atPath path: String) -> String {
|
private static func getFontName(atPath path: String) -> FontName {
|
||||||
// print("fc-scan --format %{postscriptname} \(path)")
|
// print("fc-scan --format %{postscriptname} \(path)")
|
||||||
// Get real font name
|
// Get real font name
|
||||||
let task = Shell.shell(["fc-scan", "--format", "%{postscriptname}", path])
|
let task = Shell.shell(["fc-scan", "--format", "%{postscriptname}", path])
|
||||||
|
|
||||||
guard let fontName = task.output, task.terminationStatus == 0 else {
|
guard let postscriptName = task.output, task.terminationStatus == 0 else {
|
||||||
let error = FontsToolError.fcScan(path, task.terminationStatus, task.output)
|
let error = FontsToolError.fcScan(path, task.terminationStatus, task.output)
|
||||||
print(error.description)
|
print(error.description)
|
||||||
Fonts.exit(withError: error)
|
Fonts.exit(withError: error)
|
||||||
}
|
}
|
||||||
|
|
||||||
return fontName
|
let pathURL = URL(fileURLWithPath: path)
|
||||||
|
let filename = pathURL
|
||||||
|
.deletingPathExtension()
|
||||||
|
.lastPathComponent
|
||||||
|
|
||||||
|
return FontName(
|
||||||
|
postscriptName: postscriptName,
|
||||||
|
filename: filename,
|
||||||
|
fileExtension: pathURL.pathExtension
|
||||||
|
)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@@ -8,7 +8,8 @@
|
|||||||
import Foundation
|
import Foundation
|
||||||
import ToolCore
|
import ToolCore
|
||||||
|
|
||||||
class FontPlistGenerator {
|
enum FontPlistGenerator {
|
||||||
|
|
||||||
static func generatePlistUIAppsFontContent(for fonts: [FontName], infoPlistPaths: [String]) -> String {
|
static func generatePlistUIAppsFontContent(for fonts: [FontName], infoPlistPaths: [String]) -> String {
|
||||||
let fontsToAddToPlist = fonts
|
let fontsToAddToPlist = fonts
|
||||||
.compactMap { $0 }
|
.compactMap { $0 }
|
||||||
@@ -16,26 +17,32 @@ class FontPlistGenerator {
|
|||||||
// Update each plist
|
// Update each plist
|
||||||
infoPlistPaths.forEach { infoPlist in
|
infoPlistPaths.forEach { infoPlist in
|
||||||
// Remove UIAppFonts value
|
// Remove UIAppFonts value
|
||||||
Shell.shell(launchPath: "/usr/libexec/PlistBuddy",
|
Shell.shell(
|
||||||
["-c", "delete :UIAppFonts", infoPlist])
|
launchPath: "/usr/libexec/PlistBuddy",
|
||||||
|
["-c", "delete :UIAppFonts", infoPlist]
|
||||||
|
)
|
||||||
|
|
||||||
// Add UIAppFonts empty array
|
// Add UIAppFonts empty array
|
||||||
debugPrint("Will PlistBuddy -c add :UIAppFonts array \(infoPlist)")
|
debugPrint("Will PlistBuddy -c add :UIAppFonts array \(infoPlist)")
|
||||||
Shell.shell(launchPath: "/usr/libexec/PlistBuddy",
|
Shell.shell(
|
||||||
["-c", "add :UIAppFonts array", infoPlist])
|
launchPath: "/usr/libexec/PlistBuddy",
|
||||||
|
["-c", "add :UIAppFonts array", infoPlist]
|
||||||
|
)
|
||||||
|
|
||||||
// Fill array with fonts
|
// Fill array with fonts
|
||||||
fontsToAddToPlist
|
fontsToAddToPlist
|
||||||
.forEach {
|
.forEach { fontName in
|
||||||
Shell.shell(launchPath: "/usr/libexec/PlistBuddy",
|
Shell.shell(
|
||||||
["-c", "add :UIAppFonts: string \($0)", infoPlist])
|
launchPath: "/usr/libexec/PlistBuddy",
|
||||||
|
["-c", "add :UIAppFonts: string \(fontName.filename).\(fontName.fileExtension)", infoPlist]
|
||||||
|
)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
var plistData = "<key>UIAppFonts</key>\n\t<array>\n"
|
var plistData = "<key>UIAppFonts</key>\n\t<array>\n"
|
||||||
fontsToAddToPlist
|
fontsToAddToPlist
|
||||||
.forEach {
|
.forEach { fontName in
|
||||||
plistData += "\t\t<string>\($0)</string>\n"
|
plistData += "\t\t<string>\(fontName.filename).\(fontName.fileExtension)</string>\n"
|
||||||
}
|
}
|
||||||
plistData += "\t</array>"
|
plistData += "\t</array>"
|
||||||
|
|
||||||
|
@@ -8,55 +8,81 @@
|
|||||||
import Foundation
|
import Foundation
|
||||||
import ToolCore
|
import ToolCore
|
||||||
|
|
||||||
class FontExtensionGenerator {
|
enum FontExtensionGenerator {
|
||||||
|
|
||||||
private static func getFontNameEnum(fontsNames: [String]) -> String {
|
private static func getFontNameEnum(
|
||||||
var enumDefinition = " enum FontName: String {\n"
|
fontsNames: [FontName],
|
||||||
|
visibility: ExtensionVisibility
|
||||||
|
) -> String {
|
||||||
|
var enumDefinition = " \(visibility) enum FontName: String {\n"
|
||||||
|
|
||||||
fontsNames.forEach {
|
fontsNames.forEach {
|
||||||
enumDefinition += " case \($0.fontNameSanitize) = \"\($0)\"\n"
|
enumDefinition += " case \($0.fontNameSanitize) = \"\($0.postscriptName)\"\n"
|
||||||
}
|
}
|
||||||
enumDefinition += " }\n"
|
enumDefinition += " }\n"
|
||||||
|
|
||||||
return enumDefinition
|
return enumDefinition
|
||||||
}
|
}
|
||||||
|
|
||||||
static func writeExtensionFile(fontsNames: [String],
|
static func writeExtensionFile(
|
||||||
staticVar: Bool,
|
fontsNames: [FontName],
|
||||||
|
isStatic: Bool,
|
||||||
extensionName: String,
|
extensionName: String,
|
||||||
extensionFilePath: String,
|
extensionFilePath: String,
|
||||||
isSwiftUI: Bool) {
|
isSwiftUI: Bool,
|
||||||
|
visibility: ExtensionVisibility
|
||||||
|
) {
|
||||||
// Create extension content
|
// Create extension content
|
||||||
let extensionContent = Self.getExtensionContent(fontsNames: fontsNames,
|
let extensionContent = Self.getExtensionContent(
|
||||||
staticVar: staticVar,
|
fontsNames: fontsNames,
|
||||||
|
isStatic: isStatic,
|
||||||
extensionName: extensionName,
|
extensionName: extensionName,
|
||||||
isSwiftUI: isSwiftUI)
|
isSwiftUI: isSwiftUI,
|
||||||
|
visibility: visibility
|
||||||
|
)
|
||||||
|
|
||||||
// Write content
|
// Write content
|
||||||
let extensionFilePathURL = URL(fileURLWithPath: extensionFilePath)
|
let extensionFilePathURL = URL(fileURLWithPath: extensionFilePath)
|
||||||
do {
|
do {
|
||||||
try extensionContent.write(to: extensionFilePathURL, atomically: false, encoding: .utf8)
|
try extensionContent.write(to: extensionFilePathURL, atomically: false, encoding: .utf8)
|
||||||
} catch (let error) {
|
} catch {
|
||||||
let error = FontsToolError.writeExtension(extensionFilePath, error.localizedDescription)
|
let error = FontsToolError.writeExtension(extensionFilePath, error.localizedDescription)
|
||||||
print(error.description)
|
print(error.description)
|
||||||
Fonts.exit(withError: error)
|
Fonts.exit(withError: error)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
static func getExtensionContent(fontsNames: [String],
|
static func getExtensionContent(
|
||||||
staticVar: Bool,
|
fontsNames: [FontName],
|
||||||
|
isStatic: Bool,
|
||||||
extensionName: String,
|
extensionName: String,
|
||||||
isSwiftUI: Bool) -> String {
|
isSwiftUI: Bool,
|
||||||
|
visibility: ExtensionVisibility
|
||||||
|
) -> String {
|
||||||
[
|
[
|
||||||
Self.getHeader(extensionClassname: extensionName, isSwiftUI: isSwiftUI),
|
Self.getHeader(
|
||||||
Self.getFontNameEnum(fontsNames: fontsNames),
|
extensionClassname: extensionName,
|
||||||
Self.getFontMethods(fontsNames: fontsNames, staticVar: staticVar, isSwiftUI: isSwiftUI),
|
isSwiftUI: isSwiftUI
|
||||||
|
),
|
||||||
|
Self.getFontNameEnum(
|
||||||
|
fontsNames: fontsNames,
|
||||||
|
visibility: visibility
|
||||||
|
),
|
||||||
|
Self.getFontMethods(
|
||||||
|
fontsNames: fontsNames,
|
||||||
|
isStatic: isStatic,
|
||||||
|
isSwiftUI: isSwiftUI,
|
||||||
|
visibility: visibility
|
||||||
|
),
|
||||||
Self.getFooter()
|
Self.getFooter()
|
||||||
]
|
]
|
||||||
.joined(separator: "\n")
|
.joined(separator: "\n")
|
||||||
}
|
}
|
||||||
|
|
||||||
private static func getHeader(extensionClassname: String, isSwiftUI: Bool) -> String {
|
private static func getHeader(
|
||||||
|
extensionClassname: String,
|
||||||
|
isSwiftUI: Bool
|
||||||
|
) -> String {
|
||||||
"""
|
"""
|
||||||
// Generated by ResgenSwift.\(Fonts.toolName) \(ResgenSwiftVersion)
|
// Generated by ResgenSwift.\(Fonts.toolName) \(ResgenSwiftVersion)
|
||||||
|
|
||||||
@@ -66,13 +92,22 @@ class FontExtensionGenerator {
|
|||||||
"""
|
"""
|
||||||
}
|
}
|
||||||
|
|
||||||
private static func getFontMethods(fontsNames: [FontName], staticVar: Bool, isSwiftUI: Bool) -> String {
|
private static func getFontMethods(
|
||||||
|
fontsNames: [FontName],
|
||||||
|
isStatic: Bool,
|
||||||
|
isSwiftUI: Bool,
|
||||||
|
visibility: ExtensionVisibility
|
||||||
|
) -> String {
|
||||||
let pragma = " // MARK: - Getter"
|
let pragma = " // MARK: - Getter"
|
||||||
|
|
||||||
var propertiesOrMethods: [String] = fontsNames
|
var propertiesOrMethods: [String] = fontsNames
|
||||||
.unique()
|
.unique()
|
||||||
.map {
|
.map {
|
||||||
$0.getProperty(isStatic: staticVar, isSwiftUI: isSwiftUI)
|
$0.getProperty(
|
||||||
|
isStatic: isStatic,
|
||||||
|
isSwiftUI: isSwiftUI,
|
||||||
|
visibility: visibility
|
||||||
|
)
|
||||||
}
|
}
|
||||||
|
|
||||||
propertiesOrMethods.insert(pragma, at: 0)
|
propertiesOrMethods.insert(pragma, at: 0)
|
||||||
|
@@ -6,41 +6,60 @@
|
|||||||
//
|
//
|
||||||
|
|
||||||
import Foundation
|
import Foundation
|
||||||
|
import ToolCore
|
||||||
|
|
||||||
typealias FontName = String
|
// swiftlint:disable no_grouping_extension
|
||||||
|
|
||||||
|
struct FontName: Hashable {
|
||||||
|
|
||||||
|
let postscriptName: String
|
||||||
|
let filename: String
|
||||||
|
let fileExtension: String
|
||||||
|
}
|
||||||
|
|
||||||
extension FontName {
|
extension FontName {
|
||||||
|
|
||||||
var fontNameSanitize: String {
|
var fontNameSanitize: String {
|
||||||
self.removeCharacters(from: "[]+-_")
|
postscriptName.removeCharacters(from: "[]+-_")
|
||||||
}
|
}
|
||||||
|
|
||||||
func getProperty(isStatic: Bool, isSwiftUI: Bool) -> String {
|
func getProperty(
|
||||||
if isSwiftUI {
|
isStatic: Bool,
|
||||||
if isStatic {
|
isSwiftUI: Bool,
|
||||||
return """
|
visibility: ExtensionVisibility
|
||||||
static let \(fontNameSanitize): ((_ size: CGFloat) -> Font) = { size in
|
) -> String {
|
||||||
|
switch (isSwiftUI, isStatic) {
|
||||||
|
case (true, true):
|
||||||
|
// SwiftUI, Static => let
|
||||||
|
"""
|
||||||
|
\(visibility) static let \(fontNameSanitize): ((_ size: CGFloat) -> Font) = { size in
|
||||||
Font.custom(FontName.\(fontNameSanitize).rawValue, size: size)
|
Font.custom(FontName.\(fontNameSanitize).rawValue, size: size)
|
||||||
}
|
}
|
||||||
"""
|
"""
|
||||||
}
|
|
||||||
return """
|
case (true, false):
|
||||||
func \(fontNameSanitize)(withSize size: CGFloat) -> Font {
|
// SwiftUI, Not Static => func
|
||||||
|
"""
|
||||||
|
\(visibility) func \(fontNameSanitize)(withSize size: CGFloat) -> Font {
|
||||||
Font.custom(FontName.\(fontNameSanitize).rawValue, size: size)
|
Font.custom(FontName.\(fontNameSanitize).rawValue, size: size)
|
||||||
}
|
}
|
||||||
"""
|
"""
|
||||||
}
|
|
||||||
// UIKit
|
case (false, true):
|
||||||
if isStatic {
|
// UIKit, Static => let
|
||||||
return """
|
"""
|
||||||
static let \(fontNameSanitize): ((_ size: CGFloat) -> UIFont) = { size in
|
\(visibility) static let \(fontNameSanitize): ((_ size: CGFloat) -> UIFont) = { size in
|
||||||
UIFont(name: FontName.\(fontNameSanitize).rawValue, size: size)!
|
UIFont(name: FontName.\(fontNameSanitize).rawValue, size: size)!
|
||||||
}
|
}
|
||||||
"""
|
"""
|
||||||
}
|
|
||||||
return """
|
case (false, false):
|
||||||
func \(fontNameSanitize)(withSize size: CGFloat) -> UIFont {
|
// UIKit, Not Static => func
|
||||||
|
"""
|
||||||
|
\(visibility) func \(fontNameSanitize)(withSize size: CGFloat) -> UIFont {
|
||||||
UIFont(name: FontName.\(fontNameSanitize).rawValue, size: size)!
|
UIFont(name: FontName.\(fontNameSanitize).rawValue, size: size)!
|
||||||
}
|
}
|
||||||
"""
|
"""
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
}
|
||||||
|
@@ -7,10 +7,13 @@
|
|||||||
|
|
||||||
import Foundation
|
import Foundation
|
||||||
|
|
||||||
class FontFileParser {
|
enum FontFileParser {
|
||||||
|
|
||||||
static func parse(_ inputFile: String) -> [String] {
|
static func parse(_ inputFile: String) -> [String] {
|
||||||
let inputFileContent = try! String(contentsOfFile: inputFile,
|
let inputFileContent = try! String( // swiftlint:disable:this force_try
|
||||||
encoding: .utf8)
|
contentsOfFile: inputFile,
|
||||||
|
encoding: .utf8
|
||||||
|
)
|
||||||
return inputFileContent.components(separatedBy: CharacterSet.newlines)
|
return inputFileContent.components(separatedBy: CharacterSet.newlines)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@@ -5,15 +5,15 @@
|
|||||||
// Created by Thibaut Schmitt on 30/08/2022.
|
// Created by Thibaut Schmitt on 30/08/2022.
|
||||||
//
|
//
|
||||||
|
|
||||||
import ToolCore
|
|
||||||
import Foundation
|
|
||||||
import ArgumentParser
|
import ArgumentParser
|
||||||
|
import Foundation
|
||||||
|
import ToolCore
|
||||||
|
|
||||||
struct Generate: ParsableCommand {
|
struct Generate: ParsableCommand {
|
||||||
|
|
||||||
// MARK: - CommandConfiguration
|
// MARK: - CommandConfiguration
|
||||||
|
|
||||||
static var configuration = CommandConfiguration(
|
static let configuration = CommandConfiguration(
|
||||||
abstract: "A utility to generate ressources based on a configuration file",
|
abstract: "A utility to generate ressources based on a configuration file",
|
||||||
version: ResgenSwiftVersion
|
version: ResgenSwiftVersion
|
||||||
)
|
)
|
||||||
@@ -28,7 +28,7 @@ struct Generate: ParsableCommand {
|
|||||||
|
|
||||||
// MARK: - Run
|
// MARK: - Run
|
||||||
|
|
||||||
public func run() throws {
|
func run() throws {
|
||||||
print("[\(Self.toolName)] Starting Resgen with configuration: \(options.configurationFile)")
|
print("[\(Self.toolName)] Starting Resgen with configuration: \(options.configurationFile)")
|
||||||
|
|
||||||
// Parse
|
// Parse
|
||||||
@@ -43,16 +43,20 @@ struct Generate: ParsableCommand {
|
|||||||
print()
|
print()
|
||||||
|
|
||||||
if let architecture = configuration.architecture {
|
if let architecture = configuration.architecture {
|
||||||
ArchitectureGenerator.writeArchitecture(architecture,
|
ArchitectureGenerator.writeArchitecture(
|
||||||
projectDirectory: options.projectDirectory)
|
architecture,
|
||||||
|
projectDirectory: options.projectDirectory
|
||||||
|
)
|
||||||
}
|
}
|
||||||
|
|
||||||
// Execute commands
|
// Execute commands
|
||||||
configuration.runnableConfigurations
|
configuration.runnableConfigurations
|
||||||
.forEach {
|
.forEach {
|
||||||
let begin = Date()
|
let begin = Date()
|
||||||
$0.run(projectDirectory: options.projectDirectory,
|
$0.run(
|
||||||
force: options.forceGeneration)
|
projectDirectory: options.projectDirectory,
|
||||||
|
force: options.forceGeneration
|
||||||
|
)
|
||||||
print("Took: \(Date().timeIntervalSince(begin))s\n")
|
print("Took: \(Date().timeIntervalSince(begin))s\n")
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@@ -8,8 +8,9 @@
|
|||||||
import Foundation
|
import Foundation
|
||||||
|
|
||||||
enum GenerateError: Error {
|
enum GenerateError: Error {
|
||||||
|
|
||||||
case fileNotExists(String)
|
case fileNotExists(String)
|
||||||
case invalidConfigurationFile(String)
|
case invalidConfigurationFile(String, String)
|
||||||
case commandError([String], String)
|
case commandError([String], String)
|
||||||
case writeFile(String, String)
|
case writeFile(String, String)
|
||||||
|
|
||||||
@@ -18,16 +19,15 @@ enum GenerateError: Error {
|
|||||||
case .fileNotExists(let filename):
|
case .fileNotExists(let filename):
|
||||||
return "error: [\(Generate.toolName)] File \(filename) does not exists"
|
return "error: [\(Generate.toolName)] File \(filename) does not exists"
|
||||||
|
|
||||||
case .invalidConfigurationFile(let filename):
|
case let .invalidConfigurationFile(filename, underneathErrorDescription):
|
||||||
return "error: [\(Generate.toolName)] File \(filename) is not a valid configuration file"
|
return "error: [\(Generate.toolName)] File \(filename) is not a valid configuration file. Underneath error: \(underneathErrorDescription)"
|
||||||
|
|
||||||
case .commandError(let command, let terminationStatus):
|
case let .commandError(command, terminationStatus):
|
||||||
let readableCommand = command
|
let readableCommand = command
|
||||||
.map { $0 }
|
|
||||||
.joined(separator: " ")
|
.joined(separator: " ")
|
||||||
return "error: [\(Generate.toolName)] An error occured while running command '\(readableCommand)'. Command terminate with status code: \(terminationStatus)"
|
return "error: [\(Generate.toolName)] An error occured while running command '\(readableCommand)'. Command terminate with status code: \(terminationStatus)"
|
||||||
|
|
||||||
case .writeFile(let filename, let info):
|
case let .writeFile(filename, info):
|
||||||
return "error: [\(Generate.toolName)] An error occured while writing file in \(filename): \(info)"
|
return "error: [\(Generate.toolName)] An error occured while writing file in \(filename): \(info)"
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@@ -5,12 +5,11 @@
|
|||||||
// Created by Thibaut Schmitt on 30/08/2022.
|
// Created by Thibaut Schmitt on 30/08/2022.
|
||||||
//
|
//
|
||||||
|
|
||||||
import Foundation
|
|
||||||
|
|
||||||
import Foundation
|
|
||||||
import ArgumentParser
|
import ArgumentParser
|
||||||
|
import Foundation
|
||||||
|
|
||||||
struct GenerateOptions: ParsableArguments {
|
struct GenerateOptions: ParsableArguments {
|
||||||
|
|
||||||
@Flag(name: [.customShort("f"), .customShort("F")], help: "Should force generation")
|
@Flag(name: [.customShort("f"), .customShort("F")], help: "Should force generation")
|
||||||
var forceGeneration = false
|
var forceGeneration = false
|
||||||
|
|
||||||
|
@@ -5,10 +5,11 @@
|
|||||||
// Created by Thibaut Schmitt on 18/11/2022.
|
// Created by Thibaut Schmitt on 18/11/2022.
|
||||||
//
|
//
|
||||||
|
|
||||||
import ToolCore
|
|
||||||
import Foundation
|
import Foundation
|
||||||
|
import ToolCore
|
||||||
|
|
||||||
|
enum ArchitectureGenerator {
|
||||||
|
|
||||||
struct ArchitectureGenerator {
|
|
||||||
static func writeArchitecture(_ architecture: ConfigurationArchitecture, projectDirectory: String) {
|
static func writeArchitecture(_ architecture: ConfigurationArchitecture, projectDirectory: String) {
|
||||||
// Create extension content
|
// Create extension content
|
||||||
var architectureContent = [
|
var architectureContent = [
|
||||||
@@ -30,7 +31,7 @@ struct ArchitectureGenerator {
|
|||||||
let architectureFilePathURL = URL(fileURLWithPath: "\(filePath)/\(filename)")
|
let architectureFilePathURL = URL(fileURLWithPath: "\(filePath)/\(filename)")
|
||||||
do {
|
do {
|
||||||
try architectureContent.write(to: architectureFilePathURL, atomically: false, encoding: .utf8)
|
try architectureContent.write(to: architectureFilePathURL, atomically: false, encoding: .utf8)
|
||||||
} catch (let error) {
|
} catch {
|
||||||
let error = GenerateError.writeFile(filename, error.localizedDescription)
|
let error = GenerateError.writeFile(filename, error.localizedDescription)
|
||||||
print(error.description)
|
print(error.description)
|
||||||
Generate.exit(withError: error)
|
Generate.exit(withError: error)
|
||||||
|
@@ -8,6 +8,7 @@
|
|||||||
import Foundation
|
import Foundation
|
||||||
|
|
||||||
struct ConfigurationFile: Codable, CustomDebugStringConvertible {
|
struct ConfigurationFile: Codable, CustomDebugStringConvertible {
|
||||||
|
|
||||||
var architecture: ConfigurationArchitecture?
|
var architecture: ConfigurationArchitecture?
|
||||||
var analytics: [AnalyticsConfiguration]
|
var analytics: [AnalyticsConfiguration]
|
||||||
var colors: [ColorsConfiguration]
|
var colors: [ColorsConfiguration]
|
||||||
@@ -44,10 +45,11 @@ struct ConfigurationFile: Codable, CustomDebugStringConvertible {
|
|||||||
}
|
}
|
||||||
|
|
||||||
struct ConfigurationArchitecture: Codable {
|
struct ConfigurationArchitecture: Codable {
|
||||||
|
|
||||||
let property: String
|
let property: String
|
||||||
let classname: String
|
let classname: String
|
||||||
let path: String?
|
let path: String?
|
||||||
let children: [ConfigurationArchitecture]?
|
let children: [Self]?
|
||||||
|
|
||||||
func getProperty(isStatic: Bool) -> String {
|
func getProperty(isStatic: Bool) -> String {
|
||||||
" \(isStatic ? "static " : "")let \(property) = \(classname)()"
|
" \(isStatic ? "static " : "")let \(property) = \(classname)()"
|
||||||
@@ -55,7 +57,7 @@ struct ConfigurationArchitecture: Codable {
|
|||||||
|
|
||||||
func getClass(generateStaticProperty: Bool = true) -> String {
|
func getClass(generateStaticProperty: Bool = true) -> String {
|
||||||
guard children?.isEmpty == false else {
|
guard children?.isEmpty == false else {
|
||||||
return "class \(classname) {}"
|
return "final class \(classname): Sendable {}"
|
||||||
}
|
}
|
||||||
|
|
||||||
let classDefinition = [
|
let classDefinition = [
|
||||||
@@ -81,31 +83,24 @@ struct ConfigurationArchitecture: Codable {
|
|||||||
}
|
}
|
||||||
|
|
||||||
struct AnalyticsConfiguration: Codable, CustomDebugStringConvertible {
|
struct AnalyticsConfiguration: Codable, CustomDebugStringConvertible {
|
||||||
|
|
||||||
let inputFile: String
|
let inputFile: String
|
||||||
let target: String
|
let target: String
|
||||||
let extensionOutputPath: String
|
let outputFile: String
|
||||||
let extensionName: String?
|
let visibility: String?
|
||||||
let extensionSuffix: String?
|
let staticMembers: Bool?
|
||||||
private let staticMembers: Bool?
|
|
||||||
|
|
||||||
var staticMembersOptions: Bool {
|
internal init(
|
||||||
if let staticMembers = staticMembers {
|
inputFile: String,
|
||||||
return staticMembers
|
|
||||||
}
|
|
||||||
return false
|
|
||||||
}
|
|
||||||
|
|
||||||
internal init(inputFile: String,
|
|
||||||
target: String,
|
target: String,
|
||||||
extensionOutputPath: String,
|
outputFile: String,
|
||||||
extensionName: String?,
|
visibility: String?,
|
||||||
extensionSuffix: String?,
|
staticMembers: Bool?
|
||||||
staticMembers: Bool?) {
|
) {
|
||||||
self.inputFile = inputFile
|
self.inputFile = inputFile
|
||||||
self.target = target
|
self.target = target
|
||||||
self.extensionOutputPath = extensionOutputPath
|
self.outputFile = outputFile
|
||||||
self.extensionName = extensionName
|
self.visibility = visibility
|
||||||
self.extensionSuffix = extensionSuffix
|
|
||||||
self.staticMembers = staticMembers
|
self.staticMembers = staticMembers
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -114,45 +109,47 @@ struct AnalyticsConfiguration: Codable, CustomDebugStringConvertible {
|
|||||||
Analytics configuration:
|
Analytics configuration:
|
||||||
- Input file: \(inputFile)
|
- Input file: \(inputFile)
|
||||||
- Target: \(target)
|
- Target: \(target)
|
||||||
- Extension output path: \(extensionOutputPath)
|
- Output file: \(outputFile)
|
||||||
- Extension name: \(extensionName ?? "-")
|
- Visiblity: \(visibility ?? "default")
|
||||||
- Extension suffix: \(extensionSuffix ?? "-")
|
- Static members: \(staticMembers != nil ? "\(String(describing: staticMembers))" : "default")
|
||||||
"""
|
"""
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
struct ColorsConfiguration: Codable, CustomDebugStringConvertible {
|
struct ColorsConfiguration: Codable, CustomDebugStringConvertible {
|
||||||
|
|
||||||
let inputFile: String
|
let inputFile: String
|
||||||
let style: String
|
let style: String
|
||||||
let xcassetsPath: String
|
let xcassetsPath: String
|
||||||
let extensionOutputPath: String
|
let extensionOutputPath: String?
|
||||||
let extensionName: String?
|
let extensionName: String?
|
||||||
let extensionNameSwiftUI: String?
|
let extensionNameUIKit: String?
|
||||||
let extensionSuffix: String?
|
let extensionSuffix: String?
|
||||||
private let staticMembers: Bool?
|
let visibility: String?
|
||||||
|
let assetBundle: String?
|
||||||
|
let staticMembers: Bool?
|
||||||
|
|
||||||
var staticMembersOptions: Bool {
|
internal init(
|
||||||
if let staticMembers = staticMembers {
|
inputFile: String,
|
||||||
return staticMembers
|
|
||||||
}
|
|
||||||
return false
|
|
||||||
}
|
|
||||||
|
|
||||||
internal init(inputFile: String,
|
|
||||||
style: String,
|
style: String,
|
||||||
xcassetsPath: String,
|
xcassetsPath: String,
|
||||||
extensionOutputPath: String,
|
extensionOutputPath: String?,
|
||||||
extensionName: String?,
|
extensionName: String?,
|
||||||
extensionNameSwiftUI: String?,
|
extensionNameUIKit: String?,
|
||||||
extensionSuffix: String?,
|
extensionSuffix: String?,
|
||||||
staticMembers: Bool?) {
|
visibility: String?,
|
||||||
|
assetBundle: String?,
|
||||||
|
staticMembers: Bool?
|
||||||
|
) {
|
||||||
self.inputFile = inputFile
|
self.inputFile = inputFile
|
||||||
self.style = style
|
self.style = style
|
||||||
self.xcassetsPath = xcassetsPath
|
self.xcassetsPath = xcassetsPath
|
||||||
self.extensionOutputPath = extensionOutputPath
|
self.extensionOutputPath = extensionOutputPath
|
||||||
self.extensionName = extensionName
|
self.extensionName = extensionName
|
||||||
self.extensionNameSwiftUI = extensionNameSwiftUI
|
self.extensionNameUIKit = extensionNameUIKit
|
||||||
self.extensionSuffix = extensionSuffix
|
self.extensionSuffix = extensionSuffix
|
||||||
|
self.visibility = visibility
|
||||||
|
self.assetBundle = assetBundle
|
||||||
self.staticMembers = staticMembers
|
self.staticMembers = staticMembers
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -162,43 +159,45 @@ struct ColorsConfiguration: Codable, CustomDebugStringConvertible {
|
|||||||
- Input file: \(inputFile)
|
- Input file: \(inputFile)
|
||||||
- Style: \(style)
|
- Style: \(style)
|
||||||
- Xcassets path: \(xcassetsPath)
|
- Xcassets path: \(xcassetsPath)
|
||||||
- Extension output path: \(extensionOutputPath)
|
- Extension output path: \(extensionOutputPath ?? "-")
|
||||||
- Extension name: \(extensionName ?? "-")
|
- Extension name: \(extensionName ?? "-")
|
||||||
- Extension name SwiftUI: \(extensionNameSwiftUI ?? "-")
|
- Extension name UIKit: \(extensionNameUIKit ?? "-")
|
||||||
- Extension suffix: \(extensionSuffix ?? "-")
|
- Extension suffix: \(extensionSuffix ?? "-")
|
||||||
|
- Visiblity: \(visibility ?? "default")
|
||||||
|
- Asset Bundle: \(assetBundle ?? "default")
|
||||||
|
- Static members: \(staticMembers != nil ? "\(String(describing: staticMembers))" : "default")
|
||||||
"""
|
"""
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
struct FontsConfiguration: Codable, CustomDebugStringConvertible {
|
struct FontsConfiguration: Codable, CustomDebugStringConvertible {
|
||||||
|
|
||||||
let inputFile: String
|
let inputFile: String
|
||||||
let extensionOutputPath: String
|
let extensionOutputPath: String?
|
||||||
let extensionName: String?
|
let extensionName: String?
|
||||||
let extensionNameSwiftUI: String?
|
let extensionNameUIKit: String?
|
||||||
let extensionSuffix: String?
|
let extensionSuffix: String?
|
||||||
let infoPlistPaths: String?
|
let infoPlistPaths: String?
|
||||||
private let staticMembers: Bool?
|
let visibility: String?
|
||||||
|
let staticMembers: Bool?
|
||||||
|
|
||||||
var staticMembersOptions: Bool {
|
internal init(
|
||||||
if let staticMembers = staticMembers {
|
inputFile: String,
|
||||||
return staticMembers
|
extensionOutputPath: String?,
|
||||||
}
|
|
||||||
return false
|
|
||||||
}
|
|
||||||
|
|
||||||
internal init(inputFile: String,
|
|
||||||
extensionOutputPath: String,
|
|
||||||
extensionName: String?,
|
extensionName: String?,
|
||||||
extensionNameSwiftUI: String?,
|
extensionNameUIKit: String?,
|
||||||
extensionSuffix: String?,
|
extensionSuffix: String?,
|
||||||
infoPlistPaths: String?,
|
infoPlistPaths: String?,
|
||||||
staticMembers: Bool?) {
|
visibility: String?,
|
||||||
|
staticMembers: Bool?
|
||||||
|
) {
|
||||||
self.inputFile = inputFile
|
self.inputFile = inputFile
|
||||||
self.extensionOutputPath = extensionOutputPath
|
self.extensionOutputPath = extensionOutputPath
|
||||||
self.extensionName = extensionName
|
self.extensionName = extensionName
|
||||||
self.extensionNameSwiftUI = extensionNameSwiftUI
|
self.extensionNameUIKit = extensionNameUIKit
|
||||||
self.extensionSuffix = extensionSuffix
|
self.extensionSuffix = extensionSuffix
|
||||||
self.infoPlistPaths = infoPlistPaths
|
self.infoPlistPaths = infoPlistPaths
|
||||||
|
self.visibility = visibility
|
||||||
self.staticMembers = staticMembers
|
self.staticMembers = staticMembers
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -206,44 +205,48 @@ struct FontsConfiguration: Codable, CustomDebugStringConvertible {
|
|||||||
"""
|
"""
|
||||||
Fonts configuration:
|
Fonts configuration:
|
||||||
- Input file: \(inputFile)
|
- Input file: \(inputFile)
|
||||||
- Extension output path: \(extensionOutputPath)
|
- Extension output path: \(extensionOutputPath ?? "-")
|
||||||
- Extension name: \(extensionName ?? "-")
|
- Extension name: \(extensionName ?? "-")
|
||||||
- Extension name SwiftUI: \(extensionNameSwiftUI ?? "-")
|
- Extension name UIKit: \(extensionNameUIKit ?? "-")
|
||||||
- Extension suffix: \(extensionSuffix ?? "-")
|
- Extension suffix: \(extensionSuffix ?? "-")
|
||||||
- InfoPlistPaths: \(infoPlistPaths ?? "-")
|
- InfoPlistPaths: \(infoPlistPaths ?? "-")
|
||||||
|
- Visiblity: \(visibility ?? "default")
|
||||||
|
- Static members: \(staticMembers != nil ? "\(String(describing: staticMembers))" : "default")
|
||||||
"""
|
"""
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
struct ImagesConfiguration: Codable, CustomDebugStringConvertible {
|
struct ImagesConfiguration: Codable, CustomDebugStringConvertible {
|
||||||
|
|
||||||
let inputFile: String
|
let inputFile: String
|
||||||
let xcassetsPath: String
|
let xcassetsPath: String
|
||||||
let extensionOutputPath: String
|
let extensionOutputPath: String?
|
||||||
let extensionName: String?
|
let extensionName: String?
|
||||||
let extensionNameSwiftUI: String?
|
let extensionNameUIKit: String?
|
||||||
let extensionSuffix: String?
|
let extensionSuffix: String?
|
||||||
private let staticMembers: Bool?
|
let visibility: String?
|
||||||
|
let assetBundle: String?
|
||||||
|
let staticMembers: Bool?
|
||||||
|
|
||||||
var staticMembersOptions: Bool {
|
internal init(
|
||||||
if let staticMembers = staticMembers {
|
inputFile: String,
|
||||||
return staticMembers
|
|
||||||
}
|
|
||||||
return false
|
|
||||||
}
|
|
||||||
|
|
||||||
internal init(inputFile: String,
|
|
||||||
xcassetsPath: String,
|
xcassetsPath: String,
|
||||||
extensionOutputPath: String,
|
extensionOutputPath: String?,
|
||||||
extensionName: String?,
|
extensionName: String?,
|
||||||
extensionNameSwiftUI: String?,
|
extensionNameUIKit: String?,
|
||||||
extensionSuffix: String?,
|
extensionSuffix: String?,
|
||||||
staticMembers: Bool?) {
|
visibility: String?,
|
||||||
|
assetBundle: String?,
|
||||||
|
staticMembers: Bool?
|
||||||
|
) {
|
||||||
self.inputFile = inputFile
|
self.inputFile = inputFile
|
||||||
self.xcassetsPath = xcassetsPath
|
self.xcassetsPath = xcassetsPath
|
||||||
self.extensionOutputPath = extensionOutputPath
|
self.extensionOutputPath = extensionOutputPath
|
||||||
self.extensionName = extensionName
|
self.extensionName = extensionName
|
||||||
self.extensionNameSwiftUI = extensionNameSwiftUI
|
self.extensionNameUIKit = extensionNameUIKit
|
||||||
self.extensionSuffix = extensionSuffix
|
self.extensionSuffix = extensionSuffix
|
||||||
|
self.visibility = visibility
|
||||||
|
self.assetBundle = assetBundle
|
||||||
self.staticMembers = staticMembers
|
self.staticMembers = staticMembers
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -252,39 +255,44 @@ struct ImagesConfiguration: Codable, CustomDebugStringConvertible {
|
|||||||
Images configuration:
|
Images configuration:
|
||||||
- Input file: \(inputFile)
|
- Input file: \(inputFile)
|
||||||
- Xcassets path: \(xcassetsPath)
|
- Xcassets path: \(xcassetsPath)
|
||||||
- Extension output path: \(extensionOutputPath)
|
- Extension output path: \(extensionOutputPath ?? "-")
|
||||||
- Extension name: \(extensionName ?? "-")
|
- Extension name: \(extensionName ?? "-")
|
||||||
- Extension name SwiftUI: \(extensionNameSwiftUI ?? "-")
|
- Extension name UIKit: \(extensionNameUIKit ?? "-")
|
||||||
- Extension suffix: \(extensionSuffix ?? "-")
|
- Extension suffix: \(extensionSuffix ?? "-")
|
||||||
|
- Visiblity: \(visibility ?? "default")
|
||||||
|
- Asset Bundle: \(assetBundle ?? "default")
|
||||||
|
- Static members: \(staticMembers != nil ? "\(String(describing: staticMembers))" : "default")
|
||||||
"""
|
"""
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
struct StringsConfiguration: Codable, CustomDebugStringConvertible {
|
struct StringsConfiguration: Codable, CustomDebugStringConvertible {
|
||||||
|
|
||||||
let inputFile: String
|
let inputFile: String
|
||||||
let outputPath: String
|
let outputPath: String
|
||||||
let langs: String
|
let langs: String
|
||||||
let defaultLang: String
|
let defaultLang: String
|
||||||
let extensionOutputPath: String
|
let extensionOutputPath: String?
|
||||||
let extensionName: String?
|
let extensionName: String?
|
||||||
let extensionSuffix: String?
|
let extensionSuffix: String?
|
||||||
private let staticMembers: Bool?
|
let visibility: String?
|
||||||
|
let assetBundle: String?
|
||||||
|
let staticMembers: Bool?
|
||||||
|
let xcStrings: Bool?
|
||||||
|
|
||||||
var staticMembersOptions: Bool {
|
internal init(
|
||||||
if let staticMembers = staticMembers {
|
inputFile: String,
|
||||||
return staticMembers
|
|
||||||
}
|
|
||||||
return false
|
|
||||||
}
|
|
||||||
|
|
||||||
internal init(inputFile: String,
|
|
||||||
outputPath: String,
|
outputPath: String,
|
||||||
langs: String,
|
langs: String,
|
||||||
defaultLang: String,
|
defaultLang: String,
|
||||||
extensionOutputPath: String,
|
extensionOutputPath: String?,
|
||||||
extensionName: String?,
|
extensionName: String?,
|
||||||
extensionSuffix: String?,
|
extensionSuffix: String?,
|
||||||
staticMembers: Bool?) {
|
visibility: String?,
|
||||||
|
assetBundle: String?,
|
||||||
|
staticMembers: Bool?,
|
||||||
|
xcStrings: Bool?
|
||||||
|
) {
|
||||||
self.inputFile = inputFile
|
self.inputFile = inputFile
|
||||||
self.outputPath = outputPath
|
self.outputPath = outputPath
|
||||||
self.langs = langs
|
self.langs = langs
|
||||||
@@ -292,7 +300,10 @@ struct StringsConfiguration: Codable, CustomDebugStringConvertible {
|
|||||||
self.extensionOutputPath = extensionOutputPath
|
self.extensionOutputPath = extensionOutputPath
|
||||||
self.extensionName = extensionName
|
self.extensionName = extensionName
|
||||||
self.extensionSuffix = extensionSuffix
|
self.extensionSuffix = extensionSuffix
|
||||||
|
self.visibility = visibility
|
||||||
|
self.assetBundle = assetBundle
|
||||||
self.staticMembers = staticMembers
|
self.staticMembers = staticMembers
|
||||||
|
self.xcStrings = xcStrings
|
||||||
}
|
}
|
||||||
|
|
||||||
var debugDescription: String {
|
var debugDescription: String {
|
||||||
@@ -302,39 +313,42 @@ struct StringsConfiguration: Codable, CustomDebugStringConvertible {
|
|||||||
- Output path: \(outputPath)
|
- Output path: \(outputPath)
|
||||||
- Langs: \(langs)
|
- Langs: \(langs)
|
||||||
- Default lang: \(defaultLang)
|
- Default lang: \(defaultLang)
|
||||||
- Extension output path: \(extensionOutputPath)
|
- Extension output path: \(extensionOutputPath ?? "-")
|
||||||
- Extension name: \(extensionName ?? "-")
|
- Extension name: \(extensionName ?? "-")
|
||||||
- Extension suffix: \(extensionSuffix ?? "-")
|
- Extension suffix: \(extensionSuffix ?? "-")
|
||||||
|
- Visiblity: \(visibility ?? "default")
|
||||||
|
- Asset Bundle: \(assetBundle ?? "default")
|
||||||
|
- Static members: \(staticMembers != nil ? "\(String(describing: staticMembers))" : "default")
|
||||||
|
- XC Strings: \(xcStrings != nil ? "\(String(describing: xcStrings))" : "default")
|
||||||
"""
|
"""
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
struct TagsConfiguration: Codable, CustomDebugStringConvertible {
|
struct TagsConfiguration: Codable, CustomDebugStringConvertible {
|
||||||
|
|
||||||
let inputFile: String
|
let inputFile: String
|
||||||
let lang: String
|
let lang: String
|
||||||
let extensionOutputPath: String
|
let extensionOutputPath: String
|
||||||
let extensionName: String?
|
let extensionName: String?
|
||||||
let extensionSuffix: String?
|
let extensionSuffix: String?
|
||||||
private let staticMembers: Bool?
|
let visibility: String?
|
||||||
|
let staticMembers: Bool?
|
||||||
|
|
||||||
var staticMembersOptions: Bool {
|
internal init(
|
||||||
if let staticMembers = staticMembers {
|
inputFile: String,
|
||||||
return staticMembers
|
|
||||||
}
|
|
||||||
return false
|
|
||||||
}
|
|
||||||
|
|
||||||
internal init(inputFile: String,
|
|
||||||
lang: String,
|
lang: String,
|
||||||
extensionOutputPath: String,
|
extensionOutputPath: String,
|
||||||
extensionName: String?,
|
extensionName: String?,
|
||||||
extensionSuffix: String?,
|
extensionSuffix: String?,
|
||||||
staticMembers: Bool?) {
|
visibility: String?,
|
||||||
|
staticMembers: Bool?
|
||||||
|
) {
|
||||||
self.inputFile = inputFile
|
self.inputFile = inputFile
|
||||||
self.lang = lang
|
self.lang = lang
|
||||||
self.extensionOutputPath = extensionOutputPath
|
self.extensionOutputPath = extensionOutputPath
|
||||||
self.extensionName = extensionName
|
self.extensionName = extensionName
|
||||||
self.extensionSuffix = extensionSuffix
|
self.extensionSuffix = extensionSuffix
|
||||||
|
self.visibility = visibility
|
||||||
self.staticMembers = staticMembers
|
self.staticMembers = staticMembers
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -346,6 +360,8 @@ struct TagsConfiguration: Codable, CustomDebugStringConvertible {
|
|||||||
- Extension output path: \(extensionOutputPath)
|
- Extension output path: \(extensionOutputPath)
|
||||||
- Extension name: \(extensionName ?? "-")
|
- Extension name: \(extensionName ?? "-")
|
||||||
- Extension suffix: \(extensionSuffix ?? "-")
|
- Extension suffix: \(extensionSuffix ?? "-")
|
||||||
|
- Visiblity: \(visibility ?? "default")
|
||||||
|
- Static members: \(staticMembers != nil ? "\(String(describing: staticMembers))" : "default")
|
||||||
"""
|
"""
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@@ -8,7 +8,8 @@
|
|||||||
import Foundation
|
import Foundation
|
||||||
import Yams
|
import Yams
|
||||||
|
|
||||||
class ConfigurationFileParser {
|
enum ConfigurationFileParser {
|
||||||
|
|
||||||
static func parse(_ configurationFile: String) -> ConfigurationFile {
|
static func parse(_ configurationFile: String) -> ConfigurationFile {
|
||||||
guard let data = FileManager().contents(atPath: configurationFile) else {
|
guard let data = FileManager().contents(atPath: configurationFile) else {
|
||||||
let error = GenerateError.fileNotExists(configurationFile)
|
let error = GenerateError.fileNotExists(configurationFile)
|
||||||
@@ -16,12 +17,15 @@ class ConfigurationFileParser {
|
|||||||
Generate.exit(withError: error)
|
Generate.exit(withError: error)
|
||||||
}
|
}
|
||||||
|
|
||||||
guard let configuration = try? YAMLDecoder().decode(ConfigurationFile.self, from: data) else {
|
do {
|
||||||
let error = GenerateError.invalidConfigurationFile(configurationFile)
|
return try YAMLDecoder().decode(ConfigurationFile.self, from: data)
|
||||||
|
} catch {
|
||||||
|
let error = GenerateError.invalidConfigurationFile(
|
||||||
|
configurationFile,
|
||||||
|
error.localizedDescription.description
|
||||||
|
)
|
||||||
print(error.description)
|
print(error.description)
|
||||||
Generate.exit(withError: error)
|
Generate.exit(withError: error)
|
||||||
}
|
}
|
||||||
|
|
||||||
return configuration
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@@ -8,7 +8,13 @@
|
|||||||
import Foundation
|
import Foundation
|
||||||
|
|
||||||
extension AnalyticsConfiguration: Runnable {
|
extension AnalyticsConfiguration: Runnable {
|
||||||
|
|
||||||
func run(projectDirectory: String, force: Bool) {
|
func run(projectDirectory: String, force: Bool) {
|
||||||
|
let args = getArguments(projectDirectory: projectDirectory, force: force)
|
||||||
|
Analytics.main(args)
|
||||||
|
}
|
||||||
|
|
||||||
|
func getArguments(projectDirectory: String, force: Bool) -> [String] {
|
||||||
var args = [String]()
|
var args = [String]()
|
||||||
|
|
||||||
if force {
|
if force {
|
||||||
@@ -19,25 +25,23 @@ extension AnalyticsConfiguration: Runnable {
|
|||||||
inputFile.prependIfRelativePath(projectDirectory),
|
inputFile.prependIfRelativePath(projectDirectory),
|
||||||
"--target",
|
"--target",
|
||||||
target,
|
target,
|
||||||
"--extension-output-path",
|
"--output-file",
|
||||||
extensionOutputPath.prependIfRelativePath(projectDirectory),
|
outputFile.prependIfRelativePath(projectDirectory)
|
||||||
"--static-members",
|
|
||||||
"\(staticMembersOptions)"
|
|
||||||
]
|
]
|
||||||
|
|
||||||
if let extensionName = extensionName {
|
// Add optional parameters
|
||||||
|
[
|
||||||
|
("--visibility", visibility),
|
||||||
|
("--static-members", staticMembers?.description)
|
||||||
|
].forEach { argumentName, argumentValue in
|
||||||
|
if let argumentValue {
|
||||||
args += [
|
args += [
|
||||||
"--extension-name",
|
argumentName,
|
||||||
extensionName
|
argumentValue
|
||||||
]
|
]
|
||||||
}
|
}
|
||||||
if let extensionSuffix = extensionSuffix {
|
|
||||||
args += [
|
|
||||||
"--extension-suffix",
|
|
||||||
extensionSuffix
|
|
||||||
]
|
|
||||||
}
|
}
|
||||||
|
|
||||||
Analytics.main(args)
|
return args
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@@ -8,6 +8,7 @@
|
|||||||
import Foundation
|
import Foundation
|
||||||
|
|
||||||
extension ColorsConfiguration: Runnable {
|
extension ColorsConfiguration: Runnable {
|
||||||
|
|
||||||
func run(projectDirectory: String, force: Bool) {
|
func run(projectDirectory: String, force: Bool) {
|
||||||
let args = getArguments(projectDirectory: projectDirectory, force: force)
|
let args = getArguments(projectDirectory: projectDirectory, force: force)
|
||||||
Colors.main(args)
|
Colors.main(args)
|
||||||
@@ -25,30 +26,25 @@ extension ColorsConfiguration: Runnable {
|
|||||||
"--style",
|
"--style",
|
||||||
style,
|
style,
|
||||||
"--xcassets-path",
|
"--xcassets-path",
|
||||||
xcassetsPath.prependIfRelativePath(projectDirectory),
|
xcassetsPath.prependIfRelativePath(projectDirectory)
|
||||||
"--extension-output-path",
|
|
||||||
extensionOutputPath.prependIfRelativePath(projectDirectory),
|
|
||||||
"--static-members",
|
|
||||||
"\(staticMembersOptions)"
|
|
||||||
]
|
]
|
||||||
|
|
||||||
if let extensionName = extensionName {
|
// Add optional parameters
|
||||||
|
[
|
||||||
|
("--extension-output-path", extensionOutputPath?.prependIfRelativePath(projectDirectory)),
|
||||||
|
("--extension-name", extensionName),
|
||||||
|
("--extension-name-ui-kit", extensionNameUIKit),
|
||||||
|
("--extension-suffix", extensionSuffix),
|
||||||
|
("--visibility", visibility),
|
||||||
|
("--asset-bundle", assetBundle),
|
||||||
|
("--static-members", staticMembers?.description)
|
||||||
|
].forEach { argumentName, argumentValue in
|
||||||
|
if let argumentValue {
|
||||||
args += [
|
args += [
|
||||||
"--extension-name",
|
argumentName,
|
||||||
extensionName
|
argumentValue
|
||||||
]
|
]
|
||||||
}
|
}
|
||||||
if let extensionNameSwiftUI = extensionNameSwiftUI {
|
|
||||||
args += [
|
|
||||||
"--extension-name-swift-ui",
|
|
||||||
extensionNameSwiftUI
|
|
||||||
]
|
|
||||||
}
|
|
||||||
if let extensionSuffix = extensionSuffix {
|
|
||||||
args += [
|
|
||||||
"--extension-suffix",
|
|
||||||
extensionSuffix
|
|
||||||
]
|
|
||||||
}
|
}
|
||||||
|
|
||||||
return args
|
return args
|
||||||
|
@@ -8,6 +8,7 @@
|
|||||||
import Foundation
|
import Foundation
|
||||||
|
|
||||||
extension FontsConfiguration: Runnable {
|
extension FontsConfiguration: Runnable {
|
||||||
|
|
||||||
func run(projectDirectory: String, force: Bool) {
|
func run(projectDirectory: String, force: Bool) {
|
||||||
let args = getArguments(projectDirectory: projectDirectory, force: force)
|
let args = getArguments(projectDirectory: projectDirectory, force: force)
|
||||||
Fonts.main(args)
|
Fonts.main(args)
|
||||||
@@ -21,34 +22,28 @@ extension FontsConfiguration: Runnable {
|
|||||||
}
|
}
|
||||||
|
|
||||||
args += [
|
args += [
|
||||||
inputFile.prependIfRelativePath(projectDirectory),
|
inputFile.prependIfRelativePath(projectDirectory)
|
||||||
"--extension-output-path",
|
|
||||||
extensionOutputPath.prependIfRelativePath(projectDirectory),
|
|
||||||
"--static-members",
|
|
||||||
"\(staticMembersOptions)"
|
|
||||||
]
|
]
|
||||||
|
|
||||||
if let extensionName = extensionName {
|
// Add optional parameters
|
||||||
|
[
|
||||||
|
("--extension-output-path", extensionOutputPath?.prependIfRelativePath(projectDirectory)),
|
||||||
|
("--extension-name", extensionName),
|
||||||
|
("--extension-name-ui-kit", extensionNameUIKit),
|
||||||
|
("--extension-suffix", extensionSuffix),
|
||||||
|
("--visibility", visibility),
|
||||||
|
("--static-members", staticMembers?.description)
|
||||||
|
].forEach { argumentName, argumentValue in
|
||||||
|
if let argumentValue {
|
||||||
args += [
|
args += [
|
||||||
"--extension-name",
|
argumentName,
|
||||||
extensionName
|
argumentValue
|
||||||
]
|
]
|
||||||
}
|
}
|
||||||
if let extensionNameSwiftUI = extensionNameSwiftUI {
|
|
||||||
args += [
|
|
||||||
"--extension-name-swift-ui",
|
|
||||||
extensionNameSwiftUI
|
|
||||||
]
|
|
||||||
}
|
}
|
||||||
|
|
||||||
if let extensionSuffix = extensionSuffix {
|
// Add infoPlist paths
|
||||||
args += [
|
if let infoPlistPaths {
|
||||||
"--extension-suffix",
|
|
||||||
extensionSuffix
|
|
||||||
]
|
|
||||||
}
|
|
||||||
|
|
||||||
if let infoPlistPaths = infoPlistPaths {
|
|
||||||
let adjustedPlistPaths = infoPlistPaths
|
let adjustedPlistPaths = infoPlistPaths
|
||||||
.split(separator: " ")
|
.split(separator: " ")
|
||||||
.map { String($0).prependIfRelativePath(projectDirectory) }
|
.map { String($0).prependIfRelativePath(projectDirectory) }
|
||||||
|
@@ -8,6 +8,7 @@
|
|||||||
import Foundation
|
import Foundation
|
||||||
|
|
||||||
extension ImagesConfiguration: Runnable {
|
extension ImagesConfiguration: Runnable {
|
||||||
|
|
||||||
func run(projectDirectory: String, force: Bool) {
|
func run(projectDirectory: String, force: Bool) {
|
||||||
let args = getArguments(projectDirectory: projectDirectory, force: force)
|
let args = getArguments(projectDirectory: projectDirectory, force: force)
|
||||||
Images.main(args)
|
Images.main(args)
|
||||||
@@ -23,30 +24,25 @@ extension ImagesConfiguration: Runnable {
|
|||||||
args += [
|
args += [
|
||||||
inputFile.prependIfRelativePath(projectDirectory),
|
inputFile.prependIfRelativePath(projectDirectory),
|
||||||
"--xcassets-path",
|
"--xcassets-path",
|
||||||
xcassetsPath.prependIfRelativePath(projectDirectory),
|
xcassetsPath.prependIfRelativePath(projectDirectory)
|
||||||
"--extension-output-path",
|
|
||||||
extensionOutputPath.prependIfRelativePath(projectDirectory),
|
|
||||||
"--static-members",
|
|
||||||
"\(staticMembersOptions)"
|
|
||||||
]
|
]
|
||||||
|
|
||||||
if let extensionName = extensionName {
|
// Add optional parameters
|
||||||
|
[
|
||||||
|
("--extension-output-path", extensionOutputPath?.prependIfRelativePath(projectDirectory)),
|
||||||
|
("--extension-name", extensionName),
|
||||||
|
("--extension-name-ui-kit", extensionNameUIKit),
|
||||||
|
("--extension-suffix", extensionSuffix),
|
||||||
|
("--visibility", visibility),
|
||||||
|
("--asset-bundle", assetBundle),
|
||||||
|
("--static-members", staticMembers?.description)
|
||||||
|
].forEach { argumentName, argumentValue in
|
||||||
|
if let argumentValue {
|
||||||
args += [
|
args += [
|
||||||
"--extension-name",
|
argumentName,
|
||||||
extensionName
|
argumentValue
|
||||||
]
|
]
|
||||||
}
|
}
|
||||||
if let extensionNameSwiftUI = extensionNameSwiftUI {
|
|
||||||
args += [
|
|
||||||
"--extension-name-swift-ui",
|
|
||||||
extensionNameSwiftUI
|
|
||||||
]
|
|
||||||
}
|
|
||||||
if let extensionSuffix = extensionSuffix {
|
|
||||||
args += [
|
|
||||||
"--extension-suffix",
|
|
||||||
extensionSuffix
|
|
||||||
]
|
|
||||||
}
|
}
|
||||||
|
|
||||||
return args
|
return args
|
||||||
|
@@ -8,6 +8,6 @@
|
|||||||
import Foundation
|
import Foundation
|
||||||
|
|
||||||
protocol Runnable {
|
protocol Runnable {
|
||||||
|
|
||||||
func run(projectDirectory: String, force: Bool)
|
func run(projectDirectory: String, force: Bool)
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@@ -8,7 +8,13 @@
|
|||||||
import Foundation
|
import Foundation
|
||||||
|
|
||||||
extension StringsConfiguration: Runnable {
|
extension StringsConfiguration: Runnable {
|
||||||
|
|
||||||
func run(projectDirectory: String, force: Bool) {
|
func run(projectDirectory: String, force: Bool) {
|
||||||
|
let args = getArguments(projectDirectory: projectDirectory, force: force)
|
||||||
|
Stringium.main(args)
|
||||||
|
}
|
||||||
|
|
||||||
|
func getArguments(projectDirectory: String, force: Bool) -> [String] {
|
||||||
var args = [String]()
|
var args = [String]()
|
||||||
|
|
||||||
if force {
|
if force {
|
||||||
@@ -22,27 +28,27 @@ extension StringsConfiguration: Runnable {
|
|||||||
"--langs",
|
"--langs",
|
||||||
langs,
|
langs,
|
||||||
"--default-lang",
|
"--default-lang",
|
||||||
defaultLang,
|
defaultLang
|
||||||
"--extension-output-path",
|
|
||||||
extensionOutputPath.prependIfRelativePath(projectDirectory),
|
|
||||||
"--static-members",
|
|
||||||
"\(staticMembersOptions)"
|
|
||||||
]
|
]
|
||||||
|
|
||||||
if let extensionName = extensionName {
|
// Add optional parameters
|
||||||
|
[
|
||||||
|
("--extension-output-path", extensionOutputPath?.prependIfRelativePath(projectDirectory)),
|
||||||
|
("--extension-name", extensionName),
|
||||||
|
("--extension-suffix", extensionSuffix),
|
||||||
|
("--visibility", visibility),
|
||||||
|
("--asset-bundle", assetBundle),
|
||||||
|
("--xc-strings", xcStrings?.description),
|
||||||
|
("--static-members", staticMembers?.description)
|
||||||
|
].forEach { argumentName, argumentValue in
|
||||||
|
if let argumentValue {
|
||||||
args += [
|
args += [
|
||||||
"--extension-name",
|
argumentName,
|
||||||
extensionName
|
argumentValue
|
||||||
]
|
]
|
||||||
}
|
}
|
||||||
|
|
||||||
if let extensionSuffix = extensionSuffix {
|
|
||||||
args += [
|
|
||||||
"--extension-suffix",
|
|
||||||
extensionSuffix
|
|
||||||
]
|
|
||||||
}
|
}
|
||||||
|
|
||||||
Stringium.main(args)
|
return args
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@@ -8,7 +8,13 @@
|
|||||||
import Foundation
|
import Foundation
|
||||||
|
|
||||||
extension TagsConfiguration: Runnable {
|
extension TagsConfiguration: Runnable {
|
||||||
|
|
||||||
func run(projectDirectory: String, force: Bool) {
|
func run(projectDirectory: String, force: Bool) {
|
||||||
|
let args = getArguments(projectDirectory: projectDirectory, force: force)
|
||||||
|
Tags.main(args)
|
||||||
|
}
|
||||||
|
|
||||||
|
func getArguments(projectDirectory: String, force: Bool) -> [String] {
|
||||||
var args = [String]()
|
var args = [String]()
|
||||||
|
|
||||||
if force {
|
if force {
|
||||||
@@ -20,24 +26,24 @@ extension TagsConfiguration: Runnable {
|
|||||||
"--lang",
|
"--lang",
|
||||||
lang,
|
lang,
|
||||||
"--extension-output-path",
|
"--extension-output-path",
|
||||||
extensionOutputPath.prependIfRelativePath(projectDirectory),
|
extensionOutputPath.prependIfRelativePath(projectDirectory)
|
||||||
"--static-members",
|
|
||||||
"\(staticMembersOptions)"
|
|
||||||
]
|
]
|
||||||
|
|
||||||
if let extensionName = extensionName {
|
// Add optional parameters
|
||||||
|
[
|
||||||
|
("--extension-name", extensionName),
|
||||||
|
("--extension-suffix", extensionSuffix),
|
||||||
|
("--visibility", visibility),
|
||||||
|
("--static-members", staticMembers?.description)
|
||||||
|
].forEach { argumentName, argumentValue in
|
||||||
|
if let argumentValue {
|
||||||
args += [
|
args += [
|
||||||
"--extension-name",
|
argumentName,
|
||||||
extensionName
|
argumentValue
|
||||||
]
|
]
|
||||||
}
|
}
|
||||||
if let extensionSuffix = extensionSuffix {
|
|
||||||
args += [
|
|
||||||
"--extension-suffix",
|
|
||||||
extensionSuffix
|
|
||||||
]
|
|
||||||
}
|
}
|
||||||
|
|
||||||
Tags.main(args)
|
return args
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@@ -7,7 +7,10 @@
|
|||||||
|
|
||||||
import Foundation
|
import Foundation
|
||||||
|
|
||||||
|
// swiftlint:disable force_unwrapping
|
||||||
|
|
||||||
extension FileManager {
|
extension FileManager {
|
||||||
|
|
||||||
func getAllRegularFileIn(directory: String) -> [String] {
|
func getAllRegularFileIn(directory: String) -> [String] {
|
||||||
var files = [String]()
|
var files = [String]()
|
||||||
guard let enumerator = self.enumerator(at: URL(string: directory)!, includingPropertiesForKeys: [.isRegularFileKey], options: [.skipsHiddenFiles, .skipsPackageDescendants]) else {
|
guard let enumerator = self.enumerator(at: URL(string: directory)!, includingPropertiesForKeys: [.isRegularFileKey], options: [.skipsHiddenFiles, .skipsPackageDescendants]) else {
|
||||||
|