Passed
Push — master ( 827696...d4e081 )
by Alexander
03:03
created
src/components/Debug/Exceptions/Handlers/PleasingPageHandler.php 1 patch
Indentation   +403 added lines, -403 removed lines patch added patch discarded remove patch
@@ -39,425 +39,425 @@
 block discarded – undo
39 39
  */
40 40
 class PleasingPageHandler extends MainHandler
41 41
 {
42
-	/**
43
-	 * The brand main of handler.
44
-	 * 
45
-	 * @var string $brand
46
-	 */
47
-	protected $brand = 'Lenevor Debug';
48
-
49
-	/**
50
-	 * A string identifier for a known IDE/text editor, or a closure
51
-	 * that resolves a string that can be used to open a given file
52
-	 * in an editor.
53
-	 * 
54
-	 * @var mixed $editor
55
-	 */
56
-	protected $editor;
57
-
58
-	/**
59
-	 * A list of known editor strings.
60
-	 * 
61
-	 * @var array $editors
62
-	 */
63
-	protected $editors = [
64
-		"vscode"   => "vscode://file/%file:%line",
65
-		"sublime"  => "subl://open?url=file://%file&line=%line",
66
-		"phpstorm" => "phpstorm://open?file://%file&line=%line",
67
-		"textmate" => "txmt://open?url=file://%file&line=%line",
68
-		"atom"     => "atom://core/open/file?filename=%file&line=%line",
69
-	];
42
+    /**
43
+     * The brand main of handler.
44
+     * 
45
+     * @var string $brand
46
+     */
47
+    protected $brand = 'Lenevor Debug';
48
+
49
+    /**
50
+     * A string identifier for a known IDE/text editor, or a closure
51
+     * that resolves a string that can be used to open a given file
52
+     * in an editor.
53
+     * 
54
+     * @var mixed $editor
55
+     */
56
+    protected $editor;
57
+
58
+    /**
59
+     * A list of known editor strings.
60
+     * 
61
+     * @var array $editors
62
+     */
63
+    protected $editors = [
64
+        "vscode"   => "vscode://file/%file:%line",
65
+        "sublime"  => "subl://open?url=file://%file&line=%line",
66
+        "phpstorm" => "phpstorm://open?file://%file&line=%line",
67
+        "textmate" => "txmt://open?url=file://%file&line=%line",
68
+        "atom"     => "atom://core/open/file?filename=%file&line=%line",
69
+    ];
70 70
 	
71
-	/**
72
-	 * The page title main of handler.
73
-	 * 
74
-	 * @var string $pageTitle
75
-	 */
76
-	protected $pageTitle = 'Lenevor Debug! There was an error';
71
+    /**
72
+     * The page title main of handler.
73
+     * 
74
+     * @var string $pageTitle
75
+     */
76
+    protected $pageTitle = 'Lenevor Debug! There was an error';
77 77
 	
78
-	/**
79
-	 * Fast lookup cache for known resource locations.
80
-	 * 
81
-	 * @var array $resourceCache
82
-	 */
83
-	protected $resourceCache = [];
78
+    /**
79
+     * Fast lookup cache for known resource locations.
80
+     * 
81
+     * @var array $resourceCache
82
+     */
83
+    protected $resourceCache = [];
84 84
 	
85
-	/**
86
-	 * The path to the directory containing the html error template directories.
87
-	 * 
88
-	 * @var array $searchPaths
89
-	 */
90
-	protected $searchPaths = [];
91
-
92
-	/**
93
-	 * Gets the table of data.
94
-	 * 
95
-	 * @var array $tables
96
-	 */
97
-	protected $tables = [];
85
+    /**
86
+     * The path to the directory containing the html error template directories.
87
+     * 
88
+     * @var array $searchPaths
89
+     */
90
+    protected $searchPaths = [];
91
+
92
+    /**
93
+     * Gets the table of data.
94
+     * 
95
+     * @var array $tables
96
+     */
97
+    protected $tables = [];
98 98
 	
99
-	/**
100
-	 * The template handler system.
101
-	 * 
102
-	 * @var string|object $template
103
-	 */
104
-	protected $template;	
99
+    /**
100
+     * The template handler system.
101
+     * 
102
+     * @var string|object $template
103
+     */
104
+    protected $template;	
105 105
 	
106
-	/**
107
-	 * Constructor. The PleasingPageHandler class.
108
-	 * 
109
-	 * @return void
110
-	 */
111
-	public function __construct()
112
-	{
113
-		$this->template      = new TemplateHandler;
114
-		$this->searchPaths[] = dirname(__DIR__).DIRECTORY_SEPARATOR.'Resources';
115
-	}
116
-
117
-	/**
118
-	 * Adds an editor resolver, identified by a string name, and that may be a 
119
-	 * string path, or a callable resolver.
120
-	 * 
121
-	 * @param  string            $identifier
122
-	 * @param  string|\Callable  $resolver
123
-	 * 
124
-	 * @return void
125
-	 */
126
-	public function addEditor($identifier, $resolver): void
127
-	{
128
-		$this->editors[$identifier] = $resolver;
129
-	}
130
-
131
-	/**
132
-	 * Adds an entry to the list of tables displayed in the template.
133
-	 * The expected data is a simple associative array. Any nested arrays
134
-	 * will be flattened with print_r.
135
-	 * 
136
-	 * @param  \Syscodes\Components\Contracts\Debug\Table  $table
137
-	 * 
138
-	 * @return void
139
-	 */
140
-	public function addTables(Table $table): void
141
-	{
142
-		$this->tables[] = $table;
143
-	}
106
+    /**
107
+     * Constructor. The PleasingPageHandler class.
108
+     * 
109
+     * @return void
110
+     */
111
+    public function __construct()
112
+    {
113
+        $this->template      = new TemplateHandler;
114
+        $this->searchPaths[] = dirname(__DIR__).DIRECTORY_SEPARATOR.'Resources';
115
+    }
116
+
117
+    /**
118
+     * Adds an editor resolver, identified by a string name, and that may be a 
119
+     * string path, or a callable resolver.
120
+     * 
121
+     * @param  string            $identifier
122
+     * @param  string|\Callable  $resolver
123
+     * 
124
+     * @return void
125
+     */
126
+    public function addEditor($identifier, $resolver): void
127
+    {
128
+        $this->editors[$identifier] = $resolver;
129
+    }
130
+
131
+    /**
132
+     * Adds an entry to the list of tables displayed in the template.
133
+     * The expected data is a simple associative array. Any nested arrays
134
+     * will be flattened with print_r.
135
+     * 
136
+     * @param  \Syscodes\Components\Contracts\Debug\Table  $table
137
+     * 
138
+     * @return void
139
+     */
140
+    public function addTables(Table $table): void
141
+    {
142
+        $this->tables[] = $table;
143
+    }
144 144
 	
145
-	/**
146
-	 * Gathers the variables that will be made available to the view.
147
-	 * 
148
-	 * @return  array
149
-	 */
150
-	protected function collectionVars(): array
151
-	{
152
-		$supervisor = $this->getSupervisor();
153
-		$style      = file_get_contents($this->getResource('css/debug.base.css'));
154
-		$jscript    = file_get_contents($this->getResource('js/debug.base.js'));
155
-		$tables     = array_merge($this->getDefaultTables(), $this->tables);
145
+    /**
146
+     * Gathers the variables that will be made available to the view.
147
+     * 
148
+     * @return  array
149
+     */
150
+    protected function collectionVars(): array
151
+    {
152
+        $supervisor = $this->getSupervisor();
153
+        $style      = file_get_contents($this->getResource('css/debug.base.css'));
154
+        $jscript    = file_get_contents($this->getResource('js/debug.base.js'));
155
+        $tables     = array_merge($this->getDefaultTables(), $this->tables);
156 156
 		
157
-		return [ 
158
-			'class' => explode('\\', $supervisor->getExceptionName()),
159
-			'stylesheet' => preg_replace('#[\r\n\t ]+#', ' ', $style),
160
-			'javascript' => preg_replace('#[\r\n\t ]+#', ' ', $jscript),
161
-			'header' => $this->getResource('views/partials/updown/header.php'),
162
-			'footer' => $this->getResource('views/partials/updown/footer.php'),
163
-			'info_exception' => $this->getResource('views/partials/info/info_exception.php'),
164
-			'section_stack_exception' => $this->getResource('views/partials/section_stack_exception.php'),
165
-			'section_frame' => $this->getResource('views/partials/section_frame.php'),
166
-			'frame_description' => $this->getResource('views/partials/frames/frame_description.php'),
167
-			'frame_list' => $this->getResource('views/partials/frames//frame_list.php'),
168
-			'section_code' => $this->getResource('views/partials/section_code.php'),
169
-			'code_source' => $this->getResource('views/partials/codes/code_source.php'),
170
-			'request_info' => $this->getResource('views/partials/request_info.php'),
171
-			'navigation' => $this->getResource('views/partials/details/navigation.php'),
172
-			'section_detail_context' => $this->getResource('views/partials/details/section_detail_context.php'),
173
-			'plain_exception' => Formatter::formatExceptionAsPlainText($this->getSupervisor()),
174
-			'handler' => $this,
175
-			'handlers' => $this->getDebug()->getHandlers(),
176
-			'debug' => $this->getDebug(),
177
-			'code' => $this->getExceptionCode(),
178
-			'message' => $supervisor->getExceptionMessage(),
179
-			'frames' => $this->getExceptionFrames(),
180
-			'tables' => $this->getProcessTables($tables),
181
-		];
182
-	}
157
+        return [ 
158
+            'class' => explode('\\', $supervisor->getExceptionName()),
159
+            'stylesheet' => preg_replace('#[\r\n\t ]+#', ' ', $style),
160
+            'javascript' => preg_replace('#[\r\n\t ]+#', ' ', $jscript),
161
+            'header' => $this->getResource('views/partials/updown/header.php'),
162
+            'footer' => $this->getResource('views/partials/updown/footer.php'),
163
+            'info_exception' => $this->getResource('views/partials/info/info_exception.php'),
164
+            'section_stack_exception' => $this->getResource('views/partials/section_stack_exception.php'),
165
+            'section_frame' => $this->getResource('views/partials/section_frame.php'),
166
+            'frame_description' => $this->getResource('views/partials/frames/frame_description.php'),
167
+            'frame_list' => $this->getResource('views/partials/frames//frame_list.php'),
168
+            'section_code' => $this->getResource('views/partials/section_code.php'),
169
+            'code_source' => $this->getResource('views/partials/codes/code_source.php'),
170
+            'request_info' => $this->getResource('views/partials/request_info.php'),
171
+            'navigation' => $this->getResource('views/partials/details/navigation.php'),
172
+            'section_detail_context' => $this->getResource('views/partials/details/section_detail_context.php'),
173
+            'plain_exception' => Formatter::formatExceptionAsPlainText($this->getSupervisor()),
174
+            'handler' => $this,
175
+            'handlers' => $this->getDebug()->getHandlers(),
176
+            'debug' => $this->getDebug(),
177
+            'code' => $this->getExceptionCode(),
178
+            'message' => $supervisor->getExceptionMessage(),
179
+            'frames' => $this->getExceptionFrames(),
180
+            'tables' => $this->getProcessTables($tables),
181
+        ];
182
+    }
183 183
 	
184
-	/**
185
-	 * The way in which the data sender (usually the server) can tell the recipient
186
-	 * (the browser, in general) what type of data is being sent in this case, html format tagged.
187
-	 * 
188
-	 * @return string
189
-	 */
190
-	public function contentType(): string
191
-	{
192
-		return 'text/html;charset=UTF-8';
193
-	}
194
-
195
-	/**
196
-	 * Gets the brand of project.
197
-	 * 
198
-	 * @return string
199
-	 */
200
-	public function getBrand(): string
201
-	{
202
-		return $this->brand;
203
-	}
204
-
205
-	/**
206
-	 * Returns the default tables.
207
-	 * 
208
-	 * @return \Syscodes\Components\Contracts\Debug\Table[]
209
-	 */
210
-	protected function getDefaultTables()
211
-	{
212
-		return [new ArrayTable('', $_SERVER)];
213
-	}
214
-
215
-	/**
216
-	 * Get the code of the exception that is currently being handled.
217
-	 * 
218
-	 * @return string
219
-	 */
220
-	protected function getExceptionCode()
221
-	{
222
-		$exception = $this->getException();
223
-		$code      = $exception->getCode();
224
-
225
-		if ($exception instanceof ErrorException) {
226
-			$code = Misc::translateErrorCode($exception->getSeverity());
227
-		}
228
-
229
-		return (string) $code;
230
-	}
231
-
232
-	/**
233
-	 * Get the stack trace frames of the exception that is currently being handled.
234
-	 * 
235
-	 * @return \Syscodes\Components\Debug\Engine\Supervisor;
236
-	 */
237
-	protected function getExceptionFrames()
238
-	{
239
-		$frames = $this->getSupervisor()->getFrames();
184
+    /**
185
+     * The way in which the data sender (usually the server) can tell the recipient
186
+     * (the browser, in general) what type of data is being sent in this case, html format tagged.
187
+     * 
188
+     * @return string
189
+     */
190
+    public function contentType(): string
191
+    {
192
+        return 'text/html;charset=UTF-8';
193
+    }
194
+
195
+    /**
196
+     * Gets the brand of project.
197
+     * 
198
+     * @return string
199
+     */
200
+    public function getBrand(): string
201
+    {
202
+        return $this->brand;
203
+    }
204
+
205
+    /**
206
+     * Returns the default tables.
207
+     * 
208
+     * @return \Syscodes\Components\Contracts\Debug\Table[]
209
+     */
210
+    protected function getDefaultTables()
211
+    {
212
+        return [new ArrayTable('', $_SERVER)];
213
+    }
214
+
215
+    /**
216
+     * Get the code of the exception that is currently being handled.
217
+     * 
218
+     * @return string
219
+     */
220
+    protected function getExceptionCode()
221
+    {
222
+        $exception = $this->getException();
223
+        $code      = $exception->getCode();
224
+
225
+        if ($exception instanceof ErrorException) {
226
+            $code = Misc::translateErrorCode($exception->getSeverity());
227
+        }
228
+
229
+        return (string) $code;
230
+    }
231
+
232
+    /**
233
+     * Get the stack trace frames of the exception that is currently being handled.
234
+     * 
235
+     * @return \Syscodes\Components\Debug\Engine\Supervisor;
236
+     */
237
+    protected function getExceptionFrames()
238
+    {
239
+        $frames = $this->getSupervisor()->getFrames();
240 240
 		
241
-		return $frames;
242
-	}
241
+        return $frames;
242
+    }
243 243
 	
244
-	/**
245
-	 * Gets the page title web.
246
-	 * 
247
-	 * @return string
248
-	 */
249
-	public function getPageTitle(): string
250
-	{
251
-		return $this->pageTitle;
252
-	}
253
-
254
-	/**
255
-	 * Processes an array of tables making sure everything is allright.
256
-	 * 
257
-	 * @param  \Syscodes\Components\Contracts\Debug\Table[]  $tables
258
-	 * 
259
-	 * @return array
260
-	 */
261
-	protected function getProcessTables(array $tables): array
262
-	{
263
-		$processTables = [];
264
-
265
-		foreach ($tables as $table) {
266
-			if ( ! $table instanceof Table) {
267
-				continue;
268
-			}
269
-
270
-			$label = $table->getLabel();
271
-
272
-			try {
273
-				$data = $table->getData();
274
-
275
-				if ( ! (is_array($data) || $data instanceof Traversable)) {
276
-					$data = [];
277
-				}
278
-			} catch (Exception $e) {
279
-				$data = [];
280
-			}
281
-
282
-			$processTables[$label] = $data;
283
-		}
284
-
285
-		return $processTables;
286
-	}
287
-
288
-	/**
289
-	 * Finds a resource, by its relative path, in all available search paths.
290
-	 *
291
-	 * @param  string  $resource
292
-	 * 
293
-	 * @return string
294
-	 * 
295
-	 * @throws \RuntimeException
296
-	 */
297
-	protected function getResource($resource)
298
-	{
299
-		if (isset($this->resourceCache[$resource])) {
300
-			return $this->resourceCache[$resource];
301
-		}
302
-
303
-		foreach ($this->searchPaths as $path) {
304
-			$fullPath = $path.DIRECTORY_SEPARATOR.$resource;
305
-
306
-			if (is_file($fullPath)) {
307
-				// Cache:
308
-				$this->resourceCache[$resource] = $fullPath;
309
-
310
-				return $fullPath;
311
-			}
312
-		}
313
-
314
-		throw new RuntimeException( 
315
-				"Could not find resource '{$resource}' in any resource paths.". 
316
-				"(searched: ".join(", ", $this->searchPaths).")");
317
-	}
244
+    /**
245
+     * Gets the page title web.
246
+     * 
247
+     * @return string
248
+     */
249
+    public function getPageTitle(): string
250
+    {
251
+        return $this->pageTitle;
252
+    }
253
+
254
+    /**
255
+     * Processes an array of tables making sure everything is allright.
256
+     * 
257
+     * @param  \Syscodes\Components\Contracts\Debug\Table[]  $tables
258
+     * 
259
+     * @return array
260
+     */
261
+    protected function getProcessTables(array $tables): array
262
+    {
263
+        $processTables = [];
264
+
265
+        foreach ($tables as $table) {
266
+            if ( ! $table instanceof Table) {
267
+                continue;
268
+            }
269
+
270
+            $label = $table->getLabel();
271
+
272
+            try {
273
+                $data = $table->getData();
274
+
275
+                if ( ! (is_array($data) || $data instanceof Traversable)) {
276
+                    $data = [];
277
+                }
278
+            } catch (Exception $e) {
279
+                $data = [];
280
+            }
281
+
282
+            $processTables[$label] = $data;
283
+        }
284
+
285
+        return $processTables;
286
+    }
287
+
288
+    /**
289
+     * Finds a resource, by its relative path, in all available search paths.
290
+     *
291
+     * @param  string  $resource
292
+     * 
293
+     * @return string
294
+     * 
295
+     * @throws \RuntimeException
296
+     */
297
+    protected function getResource($resource)
298
+    {
299
+        if (isset($this->resourceCache[$resource])) {
300
+            return $this->resourceCache[$resource];
301
+        }
302
+
303
+        foreach ($this->searchPaths as $path) {
304
+            $fullPath = $path.DIRECTORY_SEPARATOR.$resource;
305
+
306
+            if (is_file($fullPath)) {
307
+                // Cache:
308
+                $this->resourceCache[$resource] = $fullPath;
309
+
310
+                return $fullPath;
311
+            }
312
+        }
313
+
314
+        throw new RuntimeException( 
315
+                "Could not find resource '{$resource}' in any resource paths.". 
316
+                "(searched: ".join(", ", $this->searchPaths).")");
317
+    }
318 318
 	
319
-	/**
320
-	 * Given an exception and status code will display the error to the client.
321
-	 * 
322
-	 * @return int|null
323
-	 */
324
-	public function handle()
325
-	{	
326
-		$templatePath = $this->getResource('views/debug.layout.php');
327
-
328
-		$vars = $this->collectionVars();
319
+    /**
320
+     * Given an exception and status code will display the error to the client.
321
+     * 
322
+     * @return int|null
323
+     */
324
+    public function handle()
325
+    {	
326
+        $templatePath = $this->getResource('views/debug.layout.php');
327
+
328
+        $vars = $this->collectionVars();
329 329
 		
330
-		if (empty($vars['message'])) $vars['message'] = __('exception.noMessage');
330
+        if (empty($vars['message'])) $vars['message'] = __('exception.noMessage');
331 331
 		
332
-		$this->template->setVariables($vars);
333
-		$this->template->render($templatePath);
332
+        $this->template->setVariables($vars);
333
+        $this->template->render($templatePath);
334 334
 		
335
-		return MainHandler::QUIT;
336
-	}
337
-
338
-	/**
339
-	 * Set the editor to use to open referenced files, by a string identifier or callable
340
-	 * that will be executed for every file reference. Should return a string.
341
-	 * 
342
-	 * @example  $debug->setEditor(function($file, $line) { return "file:///{$file}"; });
343
-	 * @example  $debug->setEditor('vscode');
344
-	 * 
345
-	 * @param  string  $editor
346
-	 * 
347
-	 * @return void
348
-	 * 
349
-	 * @throws \InvalidArgumentException
350
-	 */
351
-	public function setEditor($editor)
352
-	{
353
-		if ( ! is_callable($editor) && ! isset($this->editors[$editor])) {
354
-			throw new InvalidArgumentException("Unknown editor identifier: [{$editor}]. Known editors: " .
355
-				implode(', ', array_keys($this->editors))
356
-			);
357
-		}
358
-
359
-		$this->editor = $editor;
360
-	}
361
-
362
-	/**
363
-	 * Given a string file path, and an integer file line,
364
-	 * executes the editor resolver and returns.
365
-	 * 
366
-	 * @param  string  $file
367
-	 * @param  int	   $line
368
-	 * 
369
-	 * @return string|bool
370
-	 * 
371
-	 * @throws \UnexpectedValueException
372
-	 */
373
-	public function getEditorAtHref($file, $line)
374
-	{
375
-		$editor = $this->getEditor($file, $line);
376
-
377
-		if (empty($editor))	{
378
-			return false;
379
-		}
380
-
381
-		if ( ! isset($editor['url']) || ! is_string($editor['url'])) {
382
-			throw new UnexpectedValueException(__METHOD__.'should always resolve to a string or a valid editor array');
383
-		}
384
-
385
-		$editor['url'] = str_replace("%file", rawurldecode($file), $editor['url']);
386
-		$editor['url'] = str_replace("%line", rawurldecode($line), $editor['url']);
387
-
388
-		return $editor['url'];
389
-	}
390
-
391
-	/**
392
-	 * The editor must be a valid callable function/closure.
393
-	 * 
394
-	 * @param  string  $file
395
-	 * @param  int	   $line
396
-	 * 
397
-	 * @return array
398
-	 */
399
-	protected function getEditor($file, $line): array
400
-	{
401
-		if ( ! $this->editor || ( ! is_string($this->editor) && ! is_callable($this->editor))) {
402
-			return [];
403
-		}
404
-
405
-		if (is_string($this->editor) && isset($this->editors[$this->editor]) && ! is_callable($this->editors[$this->editor])) {
406
-			return ['url' => $this->editors[$this->editor]];
407
-		}
408
-
409
-		if (is_callable($this->editor) || (isset($this->editors[$this->editor]) && is_callable($this->editors[$this->editor]))) {
410
-			if (is_callable($this->editor)) {
411
-				$callback = call_user_func($this->editor, $file, $line);
412
-			} else {
413
-				$callback = call_user_func($this->editors[$this->editor], $file, $line);
414
-			}
415
-
416
-			if (empty($callback)) {
417
-				return [];
418
-			}
419
-
420
-			if (is_string($callback)) {
421
-				return ['url' => $callback];
422
-			}
335
+        return MainHandler::QUIT;
336
+    }
337
+
338
+    /**
339
+     * Set the editor to use to open referenced files, by a string identifier or callable
340
+     * that will be executed for every file reference. Should return a string.
341
+     * 
342
+     * @example  $debug->setEditor(function($file, $line) { return "file:///{$file}"; });
343
+     * @example  $debug->setEditor('vscode');
344
+     * 
345
+     * @param  string  $editor
346
+     * 
347
+     * @return void
348
+     * 
349
+     * @throws \InvalidArgumentException
350
+     */
351
+    public function setEditor($editor)
352
+    {
353
+        if ( ! is_callable($editor) && ! isset($this->editors[$editor])) {
354
+            throw new InvalidArgumentException("Unknown editor identifier: [{$editor}]. Known editors: " .
355
+                implode(', ', array_keys($this->editors))
356
+            );
357
+        }
358
+
359
+        $this->editor = $editor;
360
+    }
361
+
362
+    /**
363
+     * Given a string file path, and an integer file line,
364
+     * executes the editor resolver and returns.
365
+     * 
366
+     * @param  string  $file
367
+     * @param  int	   $line
368
+     * 
369
+     * @return string|bool
370
+     * 
371
+     * @throws \UnexpectedValueException
372
+     */
373
+    public function getEditorAtHref($file, $line)
374
+    {
375
+        $editor = $this->getEditor($file, $line);
376
+
377
+        if (empty($editor))	{
378
+            return false;
379
+        }
380
+
381
+        if ( ! isset($editor['url']) || ! is_string($editor['url'])) {
382
+            throw new UnexpectedValueException(__METHOD__.'should always resolve to a string or a valid editor array');
383
+        }
384
+
385
+        $editor['url'] = str_replace("%file", rawurldecode($file), $editor['url']);
386
+        $editor['url'] = str_replace("%line", rawurldecode($line), $editor['url']);
387
+
388
+        return $editor['url'];
389
+    }
390
+
391
+    /**
392
+     * The editor must be a valid callable function/closure.
393
+     * 
394
+     * @param  string  $file
395
+     * @param  int	   $line
396
+     * 
397
+     * @return array
398
+     */
399
+    protected function getEditor($file, $line): array
400
+    {
401
+        if ( ! $this->editor || ( ! is_string($this->editor) && ! is_callable($this->editor))) {
402
+            return [];
403
+        }
404
+
405
+        if (is_string($this->editor) && isset($this->editors[$this->editor]) && ! is_callable($this->editors[$this->editor])) {
406
+            return ['url' => $this->editors[$this->editor]];
407
+        }
408
+
409
+        if (is_callable($this->editor) || (isset($this->editors[$this->editor]) && is_callable($this->editors[$this->editor]))) {
410
+            if (is_callable($this->editor)) {
411
+                $callback = call_user_func($this->editor, $file, $line);
412
+            } else {
413
+                $callback = call_user_func($this->editors[$this->editor], $file, $line);
414
+            }
415
+
416
+            if (empty($callback)) {
417
+                return [];
418
+            }
419
+
420
+            if (is_string($callback)) {
421
+                return ['url' => $callback];
422
+            }
423 423
 			
424
-			return ['url' => isset($callback['url']) ? $callback['url'] : $callback];
425
-		}
424
+            return ['url' => isset($callback['url']) ? $callback['url'] : $callback];
425
+        }
426 426
 		
427
-		return [];
428
-	}
429
-
430
-	/**
431
-	 * Registered the editor.
432
-	 * 
433
-	 * @return string
434
-	 */
435
-	public function getEditorcode(): string
436
-	{
437
-		return $this->editor;
438
-	}
427
+        return [];
428
+    }
429
+
430
+    /**
431
+     * Registered the editor.
432
+     * 
433
+     * @return string
434
+     */
435
+    public function getEditorcode(): string
436
+    {
437
+        return $this->editor;
438
+    }
439 439
 	
440
-	/**
441
-	 * Sets the brand of project.
442
-	 * 
443
-	 * @param  string  $brand
444
-	 * 
445
-	 * @return void
446
-	 */
447
-	public function setBrand($brand): void
448
-	{
449
-		$this->brand = (string) $brand;
450
-	}
440
+    /**
441
+     * Sets the brand of project.
442
+     * 
443
+     * @param  string  $brand
444
+     * 
445
+     * @return void
446
+     */
447
+    public function setBrand($brand): void
448
+    {
449
+        $this->brand = (string) $brand;
450
+    }
451 451
 	
452
-	/**
453
-	 * Sets the page title web.
454
-	 * 
455
-	 * @param  string  $title
456
-	 * 
457
-	 * @return void
458
-	 */
459
-	public function setPageTitle($title): void
460
-	{
461
-		$this->pageTitle = (string) $title;
462
-	}
452
+    /**
453
+     * Sets the page title web.
454
+     * 
455
+     * @param  string  $title
456
+     * 
457
+     * @return void
458
+     */
459
+    public function setPageTitle($title): void
460
+    {
461
+        $this->pageTitle = (string) $title;
462
+    }
463 463
 }
464 464
\ No newline at end of file
Please login to merge, or discard this patch.