Page Menu
Home
c4science
Search
Configure Global Search
Log In
Files
F108796846
PhutilRemarkupEngine.php
No One
Temporary
Actions
Download File
Edit File
Delete File
View Transforms
Subscribe
Mute Notifications
Award Token
Subscribers
None
File Metadata
Details
File Info
Storage
Attached
Created
Fri, Apr 18, 11:57
Size
5 KB
Mime Type
text/x-php
Expires
Sun, Apr 20, 11:57 (2 d)
Engine
blob
Format
Raw Data
Handle
25648153
Attached To
rPHU libphutil
PhutilRemarkupEngine.php
View Options
<?php
/**
* @group markup
*/
final
class
PhutilRemarkupEngine
extends
PhutilMarkupEngine
{
const
MODE_DEFAULT
=
0
;
const
MODE_TEXT
=
1
;
/**
* @var PhutilRemarkupEngineBlockRule[]
*/
private
$blockRules
=
array
();
private
$config
=
array
();
private
$mode
;
private
$metadata
=
array
();
private
$states
=
array
();
private
$postprocessRules
=
array
();
public
function
setConfig
(
$key
,
$value
)
{
$this
->
config
[
$key
]
=
$value
;
return
$this
;
}
public
function
getConfig
(
$key
,
$default
=
null
)
{
return
idx
(
$this
->
config
,
$key
,
$default
);
}
public
function
setMode
(
$mode
)
{
$this
->
mode
=
$mode
;
return
$this
;
}
public
function
isTextMode
()
{
return
$this
->
mode
&
self
::
MODE_TEXT
;
}
public
function
setBlockRules
(
array
$rules
)
{
assert_instances_of
(
$rules
,
'PhutilRemarkupEngineBlockRule'
);
$rules
=
msort
(
$rules
,
'getPriority'
);
$this
->
blockRules
=
$rules
;
foreach
(
$this
->
blockRules
as
$rule
)
{
$rule
->
setEngine
(
$this
);
}
$post_rules
=
array
();
foreach
(
$this
->
blockRules
as
$block_rule
)
{
foreach
(
$block_rule
->
getMarkupRules
()
as
$rule
)
{
$key
=
$rule
->
getPostprocessKey
();
if
(
$key
!==
null
)
{
$post_rules
[
$key
]
=
$rule
;
}
}
}
$this
->
postprocessRules
=
$post_rules
;
return
$this
;
}
public
function
getTextMetadata
(
$key
,
$default
=
null
)
{
if
(
isset
(
$this
->
metadata
[
$key
]))
{
return
$this
->
metadata
[
$key
];
}
return
idx
(
$this
->
metadata
,
$key
,
$default
);
}
public
function
setTextMetadata
(
$key
,
$value
)
{
$this
->
metadata
[
$key
]
=
$value
;
return
$this
;
}
public
function
storeText
(
$text
)
{
if
(
$this
->
isTextMode
())
{
$text
=
phutil_safe_html
(
$text
);
}
return
$this
->
storage
->
store
(
$text
);
}
public
function
overwriteStoredText
(
$token
,
$new_text
)
{
if
(
$this
->
isTextMode
())
{
$new_text
=
phutil_safe_html
(
$new_text
);
}
$this
->
storage
->
overwrite
(
$token
,
$new_text
);
return
$this
;
}
public
function
markupText
(
$text
)
{
return
$this
->
postprocessText
(
$this
->
preprocessText
(
$text
));
}
public
function
pushState
(
$state
)
{
if
(
empty
(
$this
->
states
[
$state
]))
{
$this
->
states
[
$state
]
=
0
;
}
$this
->
states
[
$state
]++;
return
$this
;
}
public
function
popState
(
$state
)
{
if
(
empty
(
$this
->
states
[
$state
]))
{
throw
new
Exception
(
"State '{$state}' pushed more than popped!"
);
}
$this
->
states
[
$state
]--;
if
(!
$this
->
states
[
$state
])
{
unset
(
$this
->
states
[
$state
]);
}
return
$this
;
}
public
function
getState
(
$state
)
{
return
!
empty
(
$this
->
states
[
$state
]);
}
public
function
preprocessText
(
$text
)
{
$this
->
metadata
=
array
();
$this
->
storage
=
new
PhutilRemarkupBlockStorage
();
// Apply basic block and paragraph normalization to the text. NOTE: We don't
// strip trailing whitespace because it is semantic in some contexts,
// notably inlined diffs that the author intends to show as a code block.
$text
=
phutil_split_lines
(
$text
,
true
);
$block_rules
=
$this
->
blockRules
;
$blocks
=
array
();
$cursor
=
0
;
$prev_block
=
array
();
while
(
isset
(
$text
[
$cursor
]))
{
$starting_cursor
=
$cursor
;
foreach
(
$block_rules
as
$block_rule
)
{
$num_lines
=
$block_rule
->
getMatchingLineCount
(
$text
,
$cursor
);
if
(
$num_lines
)
{
if
(
$blocks
)
{
$prev_block
=
last
(
$blocks
);
}
$curr_block
=
array
(
"block"
=>
implode
(
""
,
array_slice
(
$text
,
$cursor
,
$num_lines
)),
"rule"
=>
$block_rule
,
);
if
(
$prev_block
&&
$this
->
shouldMergeBlocks
(
$prev_block
,
$curr_block
))
{
$blocks
[
last_key
(
$blocks
)][
"block"
]
.=
$curr_block
[
"block"
];
}
else
{
$blocks
[]
=
$curr_block
;
}
$cursor
+=
$num_lines
;
break
;
}
}
if
(
$starting_cursor
===
$cursor
)
{
throw
new
Exception
(
"Block in text did not match any block rule."
);
}
}
$output
=
array
();
foreach
(
$blocks
as
$block
)
{
$output
[]
=
$block
[
'rule'
]->
markupText
(
$block
[
'block'
]);
}
$map
=
$this
->
storage
->
getMap
();
unset
(
$this
->
storage
);
$metadata
=
$this
->
metadata
;
if
(
$this
->
isTextMode
())
{
$output
=
implode
(
"
\n\n
"
,
$output
).
"
\n
"
;
}
else
{
$output
=
phutil_implode_html
(
"
\n\n
"
,
$output
);
}
return
array
(
'output'
=>
$output
,
'storage'
=>
$map
,
'metadata'
=>
$metadata
,
);
}
public
function
shouldMergeBlocks
(
$prev_block
,
$curr_block
)
{
$block_rules
=
ipull
(
array
(
$prev_block
,
$curr_block
),
"rule"
);
$default_rule
=
"PhutilRemarkupEngineRemarkupDefaultBlockRule"
;
try
{
assert_instances_of
(
$block_rules
,
$default_rule
);
// If the last block was empty keep merging
if
(!
strlen
(
trim
(
$prev_block
[
"block"
])))
{
return
true
;
}
// If this line is blank keep merging
if
(!
strlen
(
trim
(
$curr_block
[
"block"
])))
{
return
true
;
}
$prev_lines
=
phutil_split_lines
(
$prev_block
[
"block"
],
true
);
// If the current line and the last line have content, keep merging
if
(
strlen
(
trim
(
last
(
$prev_lines
))))
{
if
(
strlen
(
trim
(
$curr_block
[
"block"
])))
{
return
true
;
}
}
}
catch
(
Exception
$e
)
{
}
return
false
;
}
public
function
postprocessText
(
array
$dict
)
{
$this
->
metadata
=
idx
(
$dict
,
'metadata'
,
array
());
$this
->
storage
=
new
PhutilRemarkupBlockStorage
();
$this
->
storage
->
setMap
(
idx
(
$dict
,
'storage'
,
array
()));
foreach
(
$this
->
blockRules
as
$block_rule
)
{
$block_rule
->
postprocess
();
}
foreach
(
$this
->
postprocessRules
as
$rule
)
{
$rule
->
didMarkupText
();
}
return
$this
->
restoreText
(
idx
(
$dict
,
'output'
),
$this
->
isTextMode
());
}
public
function
restoreText
(
$text
)
{
return
$this
->
storage
->
restore
(
$text
,
$this
->
isTextMode
());
}
}
Event Timeline
Log In to Comment