Skip to content
Projects
Groups
Snippets
Help
This project
Loading...
Sign in / Register
Toggle navigation
Y
yii2
Project
Overview
Details
Activity
Cycle Analytics
Repository
Repository
Files
Commits
Branches
Tags
Contributors
Graph
Compare
Charts
Issues
0
Issues
0
List
Board
Labels
Milestones
Merge Requests
0
Merge Requests
0
CI / CD
CI / CD
Pipelines
Jobs
Schedules
Charts
Wiki
Wiki
Snippets
Snippets
Members
Members
Collapse sidebar
Close sidebar
Activity
Graph
Charts
Create a new issue
Jobs
Commits
Issue Boards
Open sidebar
PSDI Army
yii2
Commits
0fa19291
Commit
0fa19291
authored
Apr 02, 2014
by
Qiang Xue
Browse files
Options
Browse Files
Download
Email Patches
Plain Diff
refactored filterWhere().
parent
85bb94a6
Hide whitespace changes
Inline
Side-by-side
Showing
6 changed files
with
119 additions
and
209 deletions
+119
-209
Query.php
extensions/sphinx/Query.php
+0
-59
Query.php
framework/db/Query.php
+0
-59
QueryInterface.php
framework/db/QueryInterface.php
+23
-23
QueryTrait.php
framework/db/QueryTrait.php
+94
-66
QueryTest.php
tests/unit/extensions/sphinx/QueryTest.php
+1
-1
QueryTest.php
tests/unit/framework/db/QueryTest.php
+1
-1
No files found.
extensions/sphinx/Query.php
View file @
0fa19291
...
@@ -470,25 +470,6 @@ class Query extends Component implements QueryInterface
...
@@ -470,25 +470,6 @@ class Query extends Component implements QueryInterface
}
}
/**
/**
* Sets the WHERE part of the query ignoring empty parameters.
*
* @param string|array $condition the conditions that should be put in the WHERE part. Please refer to [[where()]]
* on how to specify this parameter.
* @param array $params the parameters (name => value) to be bound to the query.
* @return static the query object itself
* @see andFilter()
* @see orFilter()
*/
public
function
filterWhere
(
$condition
,
$params
=
[])
{
$condition
=
$this
->
filterCondition
(
$condition
);
if
(
$condition
!==
[])
{
$this
->
where
(
$condition
,
$params
);
}
return
$this
;
}
/**
* Adds an additional WHERE condition to the existing one.
* Adds an additional WHERE condition to the existing one.
* The new condition and the existing one will be joined using the 'AND' operator.
* The new condition and the existing one will be joined using the 'AND' operator.
* @param string|array $condition the new WHERE condition. Please refer to [[where()]]
* @param string|array $condition the new WHERE condition. Please refer to [[where()]]
...
@@ -510,26 +491,6 @@ class Query extends Component implements QueryInterface
...
@@ -510,26 +491,6 @@ class Query extends Component implements QueryInterface
}
}
/**
/**
* Adds an additional WHERE condition to the existing one ignoring empty parameters.
* The new condition and the existing one will be joined using the 'AND' operator.
*
* @param string|array $condition the new WHERE condition. Please refer to [[where()]]
* on how to specify this parameter.
* @param array $params the parameters (name => value) to be bound to the query.
* @return static the query object itself
* @see filter()
* @see orFilter()
*/
public
function
andFilterWhere
(
$condition
,
$params
=
[])
{
$condition
=
$this
->
filterCondition
(
$condition
);
if
(
$condition
!==
[])
{
$this
->
andWhere
(
$condition
,
$params
);
}
return
$this
;
}
/**
* Adds an additional WHERE condition to the existing one.
* Adds an additional WHERE condition to the existing one.
* The new condition and the existing one will be joined using the 'OR' operator.
* The new condition and the existing one will be joined using the 'OR' operator.
* @param string|array $condition the new WHERE condition. Please refer to [[where()]]
* @param string|array $condition the new WHERE condition. Please refer to [[where()]]
...
@@ -551,26 +512,6 @@ class Query extends Component implements QueryInterface
...
@@ -551,26 +512,6 @@ class Query extends Component implements QueryInterface
}
}
/**
/**
* Adds an additional WHERE condition to the existing one ignoring empty parameters.
* The new condition and the existing one will be joined using the 'OR' operator.
*
* @param string|array $condition the new WHERE condition. Please refer to [[where()]]
* on how to specify this parameter.
* @param array $params the parameters (name => value) to be bound to the query.
* @return static the query object itself
* @see filter()
* @see andFilter()
*/
public
function
orFilterWhere
(
$condition
,
$params
=
[])
{
$condition
=
$this
->
filterCondition
(
$condition
);
if
(
$condition
!==
[])
{
$this
->
orWhere
(
$condition
,
$params
);
}
return
$this
;
}
/**
* Sets the GROUP BY part of the query.
* Sets the GROUP BY part of the query.
* @param string|array $columns the columns to be grouped by.
* @param string|array $columns the columns to be grouped by.
* Columns can be specified in either a string (e.g. "id, name") or an array (e.g. ['id', 'name']).
* Columns can be specified in either a string (e.g. "id, name") or an array (e.g. ['id', 'name']).
...
...
framework/db/Query.php
View file @
0fa19291
...
@@ -573,65 +573,6 @@ class Query extends Component implements QueryInterface
...
@@ -573,65 +573,6 @@ class Query extends Component implements QueryInterface
}
}
/**
/**
* Sets the WHERE part of the query ignoring empty parameters.
*
* @param string|array $condition the conditions that should be put in the WHERE part. Please refer to [[where()]]
* on how to specify this parameter.
* @param array $params the parameters (name => value) to be bound to the query.
* @return static the query object itself
* @see andFilter()
* @see orFilter()
*/
public
function
filterWhere
(
$condition
,
$params
=
[])
{
$condition
=
$this
->
filterCondition
(
$condition
);
if
(
$condition
!==
[])
{
$this
->
where
(
$condition
,
$params
);
}
return
$this
;
}
/**
* Adds an additional WHERE condition to the existing one ignoring empty parameters.
* The new condition and the existing one will be joined using the 'AND' operator.
*
* @param string|array $condition the new WHERE condition. Please refer to [[where()]]
* on how to specify this parameter.
* @param array $params the parameters (name => value) to be bound to the query.
* @return static the query object itself
* @see filter()
* @see orFilter()
*/
public
function
andFilterWhere
(
$condition
,
$params
=
[])
{
$condition
=
$this
->
filterCondition
(
$condition
);
if
(
$condition
!==
[])
{
$this
->
andWhere
(
$condition
,
$params
);
}
return
$this
;
}
/**
* Adds an additional WHERE condition to the existing one ignoring empty parameters.
* The new condition and the existing one will be joined using the 'OR' operator.
*
* @param string|array $condition the new WHERE condition. Please refer to [[where()]]
* on how to specify this parameter.
* @param array $params the parameters (name => value) to be bound to the query.
* @return static the query object itself
* @see filter()
* @see andFilter()
*/
public
function
orFilterWhere
(
$condition
,
$params
=
[])
{
$condition
=
$this
->
filterCondition
(
$condition
);
if
(
$condition
!==
[])
{
$this
->
orWhere
(
$condition
,
$params
);
}
return
$this
;
}
/**
* Appends a JOIN part to the query.
* Appends a JOIN part to the query.
* The first parameter specifies what type of join it is.
* The first parameter specifies what type of join it is.
* @param string $type the type of join, such as INNER JOIN, LEFT JOIN.
* @param string $type the type of join, such as INNER JOIN, LEFT JOIN.
...
...
framework/db/QueryInterface.php
View file @
0fa19291
...
@@ -136,7 +136,7 @@ interface QueryInterface
...
@@ -136,7 +136,7 @@ interface QueryInterface
* - `or not like`: similar to the `not like` operator except that `OR` is used to concatenate
* - `or not like`: similar to the `not like` operator except that `OR` is used to concatenate
* the `NOT LIKE` predicates.
* the `NOT LIKE` predicates.
*
*
* @param array $condition the conditions that should be put in the WHERE part.
* @param
string|
array $condition the conditions that should be put in the WHERE part.
* @return static the query object itself
* @return static the query object itself
* @see andWhere()
* @see andWhere()
* @see orWhere()
* @see orWhere()
...
@@ -144,59 +144,59 @@ interface QueryInterface
...
@@ -144,59 +144,59 @@ interface QueryInterface
public
function
where
(
$condition
);
public
function
where
(
$condition
);
/**
/**
*
Sets the WHERE part of the query ignoring empty parameters
.
*
Adds an additional WHERE condition to the existing one
.
*
*
The new condition and the existing one will be joined using the 'AND' operator.
* @param
array $condition the conditions that should be put in the WHERE part
. Please refer to [[where()]]
* @param
string|array $condition the new WHERE condition
. Please refer to [[where()]]
* on how to specify this parameter.
* on how to specify this parameter.
* @return static the query object itself
* @return static the query object itself
* @see
andFilterW
here()
* @see
w
here()
* @see or
Filter
Where()
* @see orWhere()
*/
*/
public
function
filter
Where
(
$condition
);
public
function
and
Where
(
$condition
);
/**
/**
* Adds an additional WHERE condition to the existing one.
* Adds an additional WHERE condition to the existing one.
* The new condition and the existing one will be joined using the '
AND
' operator.
* The new condition and the existing one will be joined using the '
OR
' operator.
* @param string|array $condition the new WHERE condition. Please refer to [[where()]]
* @param string|array $condition the new WHERE condition. Please refer to [[where()]]
* on how to specify this parameter.
* on how to specify this parameter.
* @return static the query object itself
* @return static the query object itself
* @see where()
* @see where()
* @see
or
Where()
* @see
and
Where()
*/
*/
public
function
and
Where
(
$condition
);
public
function
or
Where
(
$condition
);
/**
/**
*
Adds an additional WHERE condition to the existing one
ignoring empty parameters.
*
Sets the WHERE part of the query
ignoring empty parameters.
*
The new condition and the existing one will be joined using the 'AND' operator.
*
* @param
string|array $condition the new WHERE condition
. Please refer to [[where()]]
* @param
array $condition the conditions that should be put in the WHERE part
. Please refer to [[where()]]
* on how to specify this parameter.
* on how to specify this parameter.
* @return static the query object itself
* @return static the query object itself
* @see
f
ilterWhere()
* @see
andF
ilterWhere()
* @see orFilterWhere()
* @see orFilterWhere()
*/
*/
public
function
andFilterWhere
(
$condition
);
public
function
filterWhere
(
array
$condition
);
/**
/**
* Adds an additional WHERE condition to the existing one.
* Adds an additional WHERE condition to the existing one
ignoring empty parameters
.
* The new condition and the existing one will be joined using the '
OR
' operator.
* The new condition and the existing one will be joined using the '
AND
' operator.
* @param
string|
array $condition the new WHERE condition. Please refer to [[where()]]
* @param array $condition the new WHERE condition. Please refer to [[where()]]
* on how to specify this parameter.
* on how to specify this parameter.
* @return static the query object itself
* @return static the query object itself
* @see
w
here()
* @see
filterW
here()
* @see
and
Where()
* @see
orFilter
Where()
*/
*/
public
function
orWhere
(
$condition
);
public
function
andFilterWhere
(
array
$condition
);
/**
/**
* Adds an additional WHERE condition to the existing one ignoring empty parameters.
* Adds an additional WHERE condition to the existing one ignoring empty parameters.
* The new condition and the existing one will be joined using the 'OR' operator.
* The new condition and the existing one will be joined using the 'OR' operator.
* @param
string|
array $condition the new WHERE condition. Please refer to [[where()]]
* @param array $condition the new WHERE condition. Please refer to [[where()]]
* on how to specify this parameter.
* on how to specify this parameter.
* @return static the query object itself
* @return static the query object itself
* @see filterWhere()
* @see filterWhere()
* @see andFilterWhere()
* @see andFilterWhere()
*/
*/
public
function
orFilterWhere
(
$condition
);
public
function
orFilterWhere
(
array
$condition
);
/**
/**
* Sets the ORDER BY part of the query.
* Sets the ORDER BY part of the query.
...
...
framework/db/QueryTrait.php
View file @
0fa19291
...
@@ -78,7 +78,7 @@ trait QueryTrait
...
@@ -78,7 +78,7 @@ trait QueryTrait
*
*
* See [[QueryInterface::where()]] for detailed documentation.
* See [[QueryInterface::where()]] for detailed documentation.
*
*
* @param array $condition the conditions that should be put in the WHERE part.
* @param
string|
array $condition the conditions that should be put in the WHERE part.
* @return static the query object itself
* @return static the query object itself
* @see andWhere()
* @see andWhere()
* @see orWhere()
* @see orWhere()
...
@@ -128,10 +128,24 @@ trait QueryTrait
...
@@ -128,10 +128,24 @@ trait QueryTrait
}
}
/**
/**
* Sets the WHERE part of the query but ignores [[isParameterNotEmpty|empty parameters]].
* Sets the WHERE part of the query but ignores [[isEmpty()|empty operands]].
*
* This method is similar to [[where()]]. The main difference is that this method will
* remove [[isEmpty()|empty query operands]]. As a result, this method is best suited
* for building query conditions based on filter values entered by users.
*
* The following code shows the difference between this method and [[where()]]:
*
*
* This function can be used to pass fields of a search form directly as search condition
* ```php
* by ignoring fields that have not been filled.
* // WHERE `age`=:age
* $query->filterWhere(['name' => null, 'age' => 20]);
* // WHERE `age`=:age
* $query->where(['age' => 20]);
* // WHERE `name` IS NULL AND `age`=:age
* $query->where(['name' => null, 'age' => 20]);
* ```
*
* Note that unlike [[where()]], you cannot pass binding parameters to this method.
*
*
* @param array $condition the conditions that should be put in the WHERE part.
* @param array $condition the conditions that should be put in the WHERE part.
* See [[where()]] on how to specify this parameter.
* See [[where()]] on how to specify this parameter.
...
@@ -140,7 +154,7 @@ trait QueryTrait
...
@@ -140,7 +154,7 @@ trait QueryTrait
* @see andFilterWhere()
* @see andFilterWhere()
* @see orFilterWhere()
* @see orFilterWhere()
*/
*/
public
function
filterWhere
(
$condition
)
public
function
filterWhere
(
array
$condition
)
{
{
$condition
=
$this
->
filterCondition
(
$condition
);
$condition
=
$this
->
filterCondition
(
$condition
);
if
(
$condition
!==
[])
{
if
(
$condition
!==
[])
{
...
@@ -150,15 +164,20 @@ trait QueryTrait
...
@@ -150,15 +164,20 @@ trait QueryTrait
}
}
/**
/**
* Adds an additional WHERE condition to the existing one but ignores [[is
ParameterNotEmpty|empty parameter
s]].
* Adds an additional WHERE condition to the existing one but ignores [[is
Empty()|empty operand
s]].
* The new condition and the existing one will be joined using the 'AND' operator.
* The new condition and the existing one will be joined using the 'AND' operator.
* @param string|array $condition the new WHERE condition. Please refer to [[where()]]
*
* This method is similar to [[andWhere()]]. The main difference is that this method will
* remove [[isEmpty()|empty query operands]]. As a result, this method is best suited
* for building query conditions based on filter values entered by users.
*
* @param array $condition the new WHERE condition. Please refer to [[where()]]
* on how to specify this parameter.
* on how to specify this parameter.
* @return static the query object itself
* @return static the query object itself
* @see filterWhere()
* @see filterWhere()
* @see orFilterWhere()
* @see orFilterWhere()
*/
*/
public
function
andFilterWhere
(
$condition
)
public
function
andFilterWhere
(
array
$condition
)
{
{
$condition
=
$this
->
filterCondition
(
$condition
);
$condition
=
$this
->
filterCondition
(
$condition
);
if
(
$condition
!==
[])
{
if
(
$condition
!==
[])
{
...
@@ -168,15 +187,20 @@ trait QueryTrait
...
@@ -168,15 +187,20 @@ trait QueryTrait
}
}
/**
/**
* Adds an additional WHERE condition to the existing one but ignores [[is
ParameterNotEmpty|empty parameter
s]].
* Adds an additional WHERE condition to the existing one but ignores [[is
Empty()|empty operand
s]].
* The new condition and the existing one will be joined using the 'OR' operator.
* The new condition and the existing one will be joined using the 'OR' operator.
* @param string|array $condition the new WHERE condition. Please refer to [[where()]]
*
* This method is similar to [[orWhere()]]. The main difference is that this method will
* remove [[isEmpty()|empty query operands]]. As a result, this method is best suited
* for building query conditions based on filter values entered by users.
*
* @param array $condition the new WHERE condition. Please refer to [[where()]]
* on how to specify this parameter.
* on how to specify this parameter.
* @return static the query object itself
* @return static the query object itself
* @see filterWhere()
* @see filterWhere()
* @see andFilterWhere()
* @see andFilterWhere()
*/
*/
public
function
orFilterWhere
(
$condition
)
public
function
orFilterWhere
(
array
$condition
)
{
{
$condition
=
$this
->
filterCondition
(
$condition
);
$condition
=
$this
->
filterCondition
(
$condition
);
if
(
$condition
!==
[])
{
if
(
$condition
!==
[])
{
...
@@ -186,88 +210,92 @@ trait QueryTrait
...
@@ -186,88 +210,92 @@ trait QueryTrait
}
}
/**
/**
* Re
turns a new condition with [[isParameterNotEmpty|empty parameters]] removed
.
* Re
moves [[isEmpty()|empty operands]] from the given query condition
.
*
*
* @param array $condition original condition
* @param array $condition
the
original condition
* @return array
condition with [[isParameterNotEmpty|empty parameter
s]] removed.
* @return array
the condition with [[isEmpty()|empty operand
s]] removed.
* @throws NotSupportedException if the condition operator is not supported
* @throws NotSupportedException if the condition operator is not supported
*/
*/
protected
function
filterCondition
(
$condition
)
protected
function
filterCondition
(
$condition
)
{
{
if
(
is_array
(
$condition
)
&&
isset
(
$condition
[
0
]))
{
if
(
!
is_array
(
$condition
))
{
$operator
=
strtoupper
(
$condition
[
0
]);
return
$condition
;
}
switch
(
$operator
)
{
if
(
!
isset
(
$condition
[
0
]))
{
case
'NOT'
:
// hash format: 'column1' => 'value1', 'column2' => 'value2', ...
case
'AND'
:
foreach
(
$condition
as
$name
=>
$value
)
{
case
'OR'
:
if
(
$this
->
isEmpty
(
$value
))
{
for
(
$i
=
1
,
$operandsCount
=
count
(
$condition
);
$i
<
$operandsCount
;
$i
++
)
{
unset
(
$condition
[
$name
]);
$subCondition
=
$this
->
filterCondition
(
$condition
[
$i
]);
}
if
(
$this
->
isParameterNotEmpty
(
$subCondition
))
{
}
$condition
[
$i
]
=
$subCondition
;
return
$condition
;
}
else
{
}
unset
(
$condition
[
$i
]);
}
// operator format: operator, operand 1, operand 2, ...
}
$operandsCount
=
count
(
$condition
)
-
1
;
$operator
=
array_shift
(
$condition
);
if
(
$operator
===
'NOT'
&&
$operandsCount
===
0
)
{
$condition
=
[];
switch
(
strtoupper
(
$operator
))
{
case
'NOT'
:
case
'AND'
:
case
'OR'
:
foreach
(
$condition
as
$i
=>
$operand
)
{
$subCondition
=
$this
->
filterCondition
(
$operand
);
if
(
$this
->
isEmpty
(
$subCondition
))
{
unset
(
$condition
[
$i
]);
}
else
{
}
else
{
// reindex array
$condition
[
$i
]
=
$subCondition
;
array_splice
(
$condition
,
0
,
0
);
if
(
$operandsCount
===
1
)
{
$condition
=
$condition
[
1
];
}
}
}
}
if
(
empty
(
$condition
))
{
return
[];
}
break
;
break
;
case
'IN'
:
case
'IN'
:
case
'NOT IN'
:
case
'NOT IN'
:
case
'LIKE'
:
case
'LIKE'
:
case
'OR LIKE'
:
case
'OR LIKE'
:
case
'NOT LIKE'
:
case
'NOT LIKE'
:
case
'OR NOT LIKE'
:
case
'OR NOT LIKE'
:
if
(
!
$this
->
isParameterNotEmpty
(
$condition
[
2
]))
{
if
(
array_key_exists
(
1
,
$condition
)
&&
$this
->
isEmpty
(
$condition
[
1
]))
{
$condition
=
[];
return
[];
}
}
break
;
break
;
case
'BETWEEN'
:
case
'BETWEEN'
:
case
'NOT BETWEEN'
:
case
'NOT BETWEEN'
:
if
(
!
$this
->
isParameterNotEmpty
(
$condition
[
2
])
&&
!
$this
->
isParameterNotEmpty
(
$condition
[
3
]))
{
if
(
array_key_exists
(
1
,
$condition
)
&&
array_key_exists
(
2
,
$condition
))
{
$condition
=
[];
if
(
$this
->
isEmpty
(
$condition
[
2
])
||
$this
->
isEmpty
(
$condition
[
3
]))
{
return
[];
}
}
}
break
;
break
;
default
:
default
:
throw
new
NotSupportedException
(
"filterWhere() does not support the '
$operator
' operator."
);
throw
new
NotSupportedException
(
"Operator not supported:
$operator
"
);
}
}
elseif
(
is_array
(
$condition
))
{
// hash format: 'column1' => 'value1', 'column2' => 'value2', ...
return
array_filter
(
$condition
,
[
$this
,
'isParameterNotEmpty'
]);
}
else
{
return
$condition
;
}
}
array_unshift
(
$condition
,
$operator
);
return
$condition
;
return
$condition
;
}
}
/**
/**
* Returns
`true` if value passed is not
"empty".
* Returns
a value indicating whether the give value is
"empty".
*
*
* The value is considered "empty", if
* The value is considered "empty", if
one of the following conditions is satisfied:
*
*
* - it is `null`,
* - it is `null`,
* - an empty string (`''`),
* - an empty string (`''`),
* - a string containing only whitespace characters,
* - a string containing only whitespace characters,
* - or an empty array.
* - or an empty array.
*
*
* @param $value
* @param
mixed
$value
* @return boolean if
parameter
is empty
* @return boolean if
the value
is empty
*/
*/
protected
function
is
ParameterNot
Empty
(
$value
)
protected
function
isEmpty
(
$value
)
{
{
if
(
is_string
(
$value
))
{
return
$value
===
''
||
$value
===
[]
||
$value
===
null
||
is_string
(
$value
)
&&
trim
(
$value
)
===
''
;
$value
=
trim
(
$value
);
}
return
$value
!==
''
&&
$value
!==
[]
&&
$value
!==
null
;
}
}
/**
/**
...
...
tests/unit/extensions/sphinx/QueryTest.php
View file @
0fa19291
...
@@ -115,7 +115,7 @@ class QueryTest extends SphinxTestCase
...
@@ -115,7 +115,7 @@ class QueryTest extends SphinxTestCase
{
{
$query
=
new
Query
();
$query
=
new
Query
();
$query
->
filterWhere
([
'and'
,
[
'like'
,
'name'
,
''
],
[
'like'
,
'title'
,
''
],
[
'id'
=>
1
],
[
'not'
,
[
'like'
,
'name'
,
''
]]]);
$query
->
filterWhere
([
'and'
,
[
'like'
,
'name'
,
''
],
[
'like'
,
'title'
,
''
],
[
'id'
=>
1
],
[
'not'
,
[
'like'
,
'name'
,
''
]]]);
$this
->
assertEquals
([
'
id'
=>
1
],
$query
->
where
);
$this
->
assertEquals
([
'
and'
,
[
'id'
=>
1
]
],
$query
->
where
);
}
}
public
function
testGroup
()
public
function
testGroup
()
...
...
tests/unit/framework/db/QueryTest.php
View file @
0fa19291
...
@@ -104,7 +104,7 @@ class QueryTest extends DatabaseTestCase
...
@@ -104,7 +104,7 @@ class QueryTest extends DatabaseTestCase
{
{
$query
=
new
Query
();
$query
=
new
Query
();
$query
->
filterWhere
([
'and'
,
[
'like'
,
'name'
,
''
],
[
'like'
,
'title'
,
''
],
[
'id'
=>
1
],
[
'not'
,
[
'like'
,
'name'
,
''
]]]);
$query
->
filterWhere
([
'and'
,
[
'like'
,
'name'
,
''
],
[
'like'
,
'title'
,
''
],
[
'id'
=>
1
],
[
'not'
,
[
'like'
,
'name'
,
''
]]]);
$this
->
assertEquals
([
'
id'
=>
1
],
$query
->
where
);
$this
->
assertEquals
([
'
and'
,
[
'id'
=>
1
]
],
$query
->
where
);
}
}
public
function
testJoin
()
public
function
testJoin
()
...
...
Write
Preview
Markdown
is supported
0%
Try again
or
attach a new file
Attach a file
Cancel
You are about to add
0
people
to the discussion. Proceed with caution.
Finish editing this message first!
Cancel
Please
register
or
sign in
to comment