Skip to content

Commit 5241395

Browse files
committed
feat: add partial support for property hooks
This commit adds support for non-abstract property hooks, meaning it allows for the creation of property hooks in classes, but not in interfaces. Additionally, property hooks with no body are not allowed. Signed-off-by: azjezz <[email protected]>
1 parent 4b56670 commit 5241395

9 files changed

+288
-17
lines changed

examples/class.rs

+76
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,76 @@
1+
use php_codegen::class::Class;
2+
use php_codegen::data_type::DataType;
3+
use php_codegen::file::File;
4+
use php_codegen::literal::Value;
5+
use php_codegen::modifiers::VisibilityModifier;
6+
use php_codegen::property::Property;
7+
use php_codegen::property::PropertyHook;
8+
use php_codegen::property::PropertySetHookParameter;
9+
10+
fn main() {
11+
let file = File::new()
12+
.namespaced("App")
13+
.declare("strict_types", 1)
14+
// A class that uses property hooks
15+
.class(
16+
Class::new("SimpleUser")
17+
.property(
18+
Property::new("firstName")
19+
.typed(DataType::String)
20+
.visibility(VisibilityModifier::Private)
21+
.default(Value::String("Jane".to_string())),
22+
)
23+
.property(
24+
Property::new("lastName")
25+
.typed(DataType::String)
26+
.visibility(VisibilityModifier::Private)
27+
.default(Value::String("Doe".to_string())),
28+
)
29+
.property(
30+
Property::new("fullname")
31+
.typed(DataType::String)
32+
.visibility(VisibilityModifier::Public)
33+
.hook(PropertyHook::Get(
34+
false,
35+
vec!["return $this->firstName . ' ' . $this->lastName;"].into(),
36+
))
37+
.hook(PropertyHook::Set(
38+
Some(
39+
PropertySetHookParameter::new("$fullname").typed(DataType::String),
40+
),
41+
vec![
42+
"[$first, $last] = explode(' ', $fullname);",
43+
"$this->firstName = $first;",
44+
"$this->lastName = $last;",
45+
]
46+
.into(),
47+
)),
48+
),
49+
)
50+
.class(
51+
Class::new("SimpleUser2")
52+
.property(
53+
Property::new("firstName")
54+
.typed(DataType::String)
55+
.visibility(VisibilityModifier::Private)
56+
.default(Value::String("Jane".to_string())),
57+
)
58+
.property(
59+
Property::new("lastName")
60+
.typed(DataType::String)
61+
.visibility(VisibilityModifier::Private)
62+
.default(Value::String("Doe".to_string())),
63+
)
64+
.property(
65+
Property::new("fullname")
66+
.typed(DataType::String)
67+
.visibility(VisibilityModifier::Public)
68+
.hook(PropertyHook::Get(
69+
false,
70+
vec!["return $this->firstName . ' ' . $this->lastName;"].into(),
71+
)),
72+
),
73+
);
74+
75+
print!("{file}");
76+
}

src/body.rs

+1-1
Original file line numberDiff line numberDiff line change
@@ -1,4 +1,4 @@
1-
use core::fmt::Debug;
1+
use std::fmt::Debug;
22

33
use crate::Generator;
44
use crate::Indentation;

src/comment.rs

+7-1
Original file line numberDiff line numberDiff line change
@@ -84,7 +84,13 @@ impl Generator for Document {
8484
impl Generator for Element {
8585
fn generate(&self, _: Indentation, _: usize) -> String {
8686
match self {
87-
Element::Tag(tag, description) => format!("@{} {}", tag, description),
87+
Element::Tag(tag, description) => {
88+
if description.is_empty() {
89+
format!("@{}", tag)
90+
} else {
91+
format!("@{} {}", tag, description)
92+
}
93+
}
8894
Element::Text(text) => text.to_string(),
8995
Element::EmptyLine => String::new(),
9096
}

src/data_type.rs

+1-1
Original file line numberDiff line numberDiff line change
@@ -31,7 +31,7 @@ impl Generator for DataType {
3131
match self {
3232
DataType::Named(name) => name.to_string(),
3333
DataType::Nullable(inner) => {
34-
format!("?{}", inner.generate(_indentation, _level))
34+
format!("null|{}", inner.generate(_indentation, _level))
3535
}
3636
DataType::Union(inner) => inner
3737
.iter()

src/interface.rs

+5-1
Original file line numberDiff line numberDiff line change
@@ -1,4 +1,8 @@
1-
use crate::{attribute::AttributeGroup, comment::Document, method::Method, Generator, Indentation};
1+
use crate::attribute::AttributeGroup;
2+
use crate::comment::Document;
3+
use crate::method::Method;
4+
use crate::Generator;
5+
use crate::Indentation;
26

37
#[derive(Debug)]
48
pub struct Interface {

src/method.rs

+1-1
Original file line numberDiff line numberDiff line change
@@ -1,4 +1,4 @@
1-
use core::fmt::Debug;
1+
use std::fmt::Debug;
22

33
use crate::attribute::AttributeGroup;
44
use crate::body::Body;

src/property.rs

+109-6
Original file line numberDiff line numberDiff line change
@@ -1,4 +1,5 @@
11
use crate::attribute::AttributeGroup;
2+
use crate::body::Body;
23
use crate::comment::Document;
34
use crate::data_type::DataType;
45
use crate::literal::Value;
@@ -7,15 +8,43 @@ use crate::modifiers::VisibilityModifier;
78
use crate::Generator;
89
use crate::Indentation;
910

11+
#[derive(Debug)]
12+
pub struct PropertySetHookParameter {
13+
pub data_type: Option<DataType>,
14+
pub name: String,
15+
}
16+
17+
#[derive(Debug)]
18+
pub enum PropertyHook {
19+
Get(bool, Body),
20+
Set(Option<PropertySetHookParameter>, Body),
21+
}
22+
1023
#[derive(Debug)]
1124
pub struct Property {
1225
pub documentation: Option<Document>,
1326
pub attributes: Vec<AttributeGroup>,
14-
pub name: String,
27+
pub visibility: Option<VisibilityModifier>,
28+
pub modifiers: Vec<Modifier>,
1529
pub data_type: Option<DataType>,
30+
pub name: String,
1631
pub default: Option<Value>,
17-
pub modifiers: Vec<Modifier>,
18-
pub visibility: Option<VisibilityModifier>,
32+
pub hooks: Vec<PropertyHook>,
33+
}
34+
35+
impl PropertySetHookParameter {
36+
pub fn new<T: ToString>(name: T) -> Self {
37+
Self {
38+
name: name.to_string(),
39+
data_type: None,
40+
}
41+
}
42+
43+
pub fn typed(mut self, data_type: DataType) -> Self {
44+
self.data_type = Some(data_type);
45+
46+
self
47+
}
1948
}
2049

2150
impl Property {
@@ -28,6 +57,7 @@ impl Property {
2857
attributes: vec![],
2958
visibility: None,
3059
documentation: None,
60+
hooks: vec![],
3161
}
3262
}
3363

@@ -84,6 +114,79 @@ impl Property {
84114

85115
self
86116
}
117+
118+
pub fn hook(mut self, hook: PropertyHook) -> Self {
119+
self.hooks.push(hook);
120+
121+
self
122+
}
123+
}
124+
125+
impl Generator for PropertySetHookParameter {
126+
fn generate(&self, indentation: Indentation, level: usize) -> String {
127+
let mut code = String::new();
128+
129+
if let Some(data_type) = &self.data_type {
130+
code.push_str(&format!("{} ", data_type.generate(indentation, level)));
131+
}
132+
133+
code.push_str(&self.name);
134+
135+
code
136+
}
137+
}
138+
139+
impl Generator for PropertyHook {
140+
fn generate(&self, indentation: Indentation, level: usize) -> String {
141+
match self {
142+
PropertyHook::Get(by_reference, body) => {
143+
let mut code = String::new();
144+
145+
code.push_str(&indentation.value(level + 1));
146+
if *by_reference {
147+
code.push_str("&get");
148+
} else {
149+
code.push_str("get");
150+
}
151+
152+
code.push_str(&body.generate(indentation, level + 1));
153+
154+
code
155+
}
156+
PropertyHook::Set(parameter, body) => {
157+
let mut code = String::new();
158+
159+
code.push_str(&indentation.value(level + 1));
160+
code.push_str("set");
161+
162+
if let Some(parameter) = parameter {
163+
code.push_str(" (");
164+
code.push_str(&parameter.generate(indentation, level + 1));
165+
code.push(')');
166+
}
167+
168+
code.push_str(&body.generate(indentation, level + 1));
169+
170+
code
171+
}
172+
}
173+
}
174+
}
175+
176+
impl Generator for Vec<PropertyHook> {
177+
fn generate(&self, indentation: Indentation, level: usize) -> String {
178+
if self.is_empty() {
179+
return String::from(";");
180+
}
181+
182+
let hooks = self
183+
.iter()
184+
.map(|hook| hook.generate(indentation, level))
185+
.collect::<Vec<String>>()
186+
.join("");
187+
188+
format!(" {{\n{}{}}}", hooks, indentation.value(level))
189+
}
87190
}
88191

89192
impl Generator for Property {
@@ -122,11 +225,11 @@ impl Generator for Property {
122225
code.push_str(&format!("${}", &self.name));
123226

124227
if let Some(default) = &self.default {
125-
code.push_str(&format!(" = {};\n", default.generate(indentation, level)));
126-
} else {
127-
code.push_str(";\n");
228+
code.push_str(&format!(" = {}", default.generate(indentation, level)));
128229
}
129230

231+
code.push_str(&self.hooks.generate(indentation, level));
232+
130233
code
131234
}
132235
}

tests/complete.php

+27-6
Original file line numberDiff line numberDiff line change
@@ -92,11 +92,8 @@ abstract class Example extends Foo\Bar\Baz implements Foo\Bar\BazInterface
9292
public const bool E = false;
9393

9494
private string $foo;
95-
9695
protected string $bar;
97-
9896
public string|int $baz = "Hello World!";
99-
10097
/**
10198
* This is a simple hello function.
10299
*
@@ -138,6 +135,33 @@ public final function helloWorld(): void {
138135
}
139136
}
140137

138+
class SimpleUser
139+
{
140+
private string $firstName = "Jane";
141+
private string $lastName = "Doe";
142+
public string $fullname {
143+
get {
144+
return $this->firstName . ' ' . $this->lastName;
145+
}
146+
set (string $fullname) {
147+
[$first, $last] = explode(' ', $fullname);
148+
$this->firstName = $first;
149+
$this->lastName = $last;
150+
}
151+
}
152+
}
153+
154+
class SimpleUser2
155+
{
156+
private string $firstName = "Jane";
157+
private string $lastName = "Doe";
158+
public string $fullname {
159+
get {
160+
return $this->firstName . ' ' . $this->lastName;
161+
}
162+
}
163+
}
164+
141165
/**
142166
* This is an example trait.
143167
*/
@@ -157,11 +181,8 @@ trait ExampleTrait
157181
}
158182

159183
private string $foo;
160-
161184
protected string $bar;
162-
163185
public string|int $baz = "Hello World!";
164-
165186
/**
166187
* This is a simple hello function.
167188
*

0 commit comments

Comments
 (0)