Skip to content

Commit 73994c5

Browse files
committed
initial commit
0 parents  commit 73994c5

9 files changed

+388
-0
lines changed

.gitignore

+2
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,2 @@
1+
vendor
2+
composer.lock

LICENSE.md

+21
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,21 @@
1+
# The MIT License (MIT)
2+
3+
Copyright (c) 2016 Malte Goldenbaum <github@webklex.com>
4+
5+
> Permission is hereby granted, free of charge, to any person obtaining a copy
6+
> of this software and associated documentation files (the "Software"), to deal
7+
> in the Software without restriction, including without limitation the rights
8+
> to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
9+
> copies of the Software, and to permit persons to whom the Software is
10+
> furnished to do so, subject to the following conditions:
11+
>
12+
> The above copyright notice and this permission notice shall be included in
13+
> all copies or substantial portions of the Software.
14+
>
15+
> THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
16+
> IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
17+
> FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
18+
> AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
19+
> LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
20+
> OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN
21+
> THE SOFTWARE.

README.md

+81
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,81 @@
1+
# Laravel PDF Merger
2+
3+
PDF merger for Laravel inspired by another package, created for personal use. Tested with Laravel 5.6.
4+
5+
## Advantages
6+
* Also works with PDF versions above `1.4`
7+
* Works with `PHP 7`
8+
9+
## Installation
10+
```bash
11+
$ composer require grofgraf/pdf-merger
12+
```
13+
14+
## Configuration
15+
```php
16+
'providers' => [
17+
...
18+
GrofGraf\PDFMerger\Providers\PDFMergerServiceProvider::class
19+
],
20+
21+
'aliases' => [
22+
...
23+
'PDFMerger' => GrofGraf\PDFMerger\Facades\PDFMergerFacade::class
24+
]
25+
```
26+
27+
> When merging PDFs versions above 1.4 or PDF strings, a temporary PDF will be created during the process and stored in `storage/tmp` directory
28+
29+
30+
31+
## Usage
32+
33+
You can add PDFs for merging, by specifying a file path of PDF with `addPathToPDF` method, or adding PDF file as string with `addPDFString` method. The second argument of both methods is array of selected pages (`'all'` for all pages) and the third argument is PDFs orientation (portrait or landscape).
34+
```php
35+
$merger->addPathToPDF('/path/to/pdf', 'all', 'P');
36+
$merger->addPDFString(file_get_contents('path/to/pdf'), ['1', '2'], 'L')
37+
```
38+
39+
You can set a merged PDF name by using `setFileName` method.
40+
```php
41+
$merger->setFileName('merger.pdf');
42+
```
43+
44+
At the end finnish process with `merge` method and use one of the output options for the merged PDF.
45+
Available output options are:
46+
* `inline()`
47+
* `download()`
48+
* `string()`
49+
* `save('path/to/merged.pdf')`
50+
51+
```php
52+
$merger->merge();
53+
$merger->inline();
54+
```
55+
56+
Example usage
57+
```php
58+
$merger = \PDFMerger::init();
59+
$merger->addPathToPDF(base_path('/vendor/grofgraf/pdf-merger/examples/one.pdf'), [2], 'P');
60+
$merger->addPDFString(file_get_contents(base_path('/vendor/grofgraf/pdf-merger/examples/two.pdf')), 'all', 'L');
61+
$merger->merge();
62+
$merger->save(base_path('/public/pdfs/merged.pdf'));
63+
```
64+
65+
## Authors
66+
* [GrofGraf](https://github.com/GrofGraf)
67+
68+
69+
## Credits
70+
* **Webklex** [LaravelPDFMerger](https://github.com/Webklex/laravel-pdfmerger)
71+
72+
## License
73+
The MIT License (MIT)
74+
75+
Copyright (c) 2017 GrofGraf
76+
77+
Permission is hereby granted, free of charge, to any person obtaining a copy of this software and associated documentation files (the "Software"), to deal in the Software without restriction, including without limitation the rights to use, copy, modify, merge, publish, distribute, sublicense, and/or sell copies of the Software, and to permit persons to whom the Software is furnished to do so, subject to the following conditions:
78+
79+
The above copyright notice and this permission notice shall be included in all copies or substantial portions of the Software.
80+
81+
THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE.

composer.json

+28
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,28 @@
1+
{
2+
"name": "grofgraf/pdf-merger",
3+
"type": "library",
4+
"description": "Package for Laravel that merges multiple PDFs into one.",
5+
"keywords": [
6+
"grofgraf",
7+
"laravel",
8+
"pdf",
9+
"merger",
10+
"pdfmerger",
11+
"pdf-merger"
12+
],
13+
"license": "MIT",
14+
"authors": [
15+
{
16+
"name": "Grof Graf"
17+
}
18+
],
19+
"require": {
20+
"illuminate/support": "^5.7.4",
21+
"setasign/fpdi-fpdf": "^2.0"
22+
},
23+
"autoload": {
24+
"psr-4": {
25+
"GrofGraf\\PDFMerger\\": "src"
26+
}
27+
}
28+
}

examples/one.pdf

18 KB
Binary file not shown.

examples/two.pdf

18 KB
Binary file not shown.

src/PDFMerger.php

+210
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,210 @@
1+
<?php
2+
3+
namespace GrofGraf\PDFMerger;
4+
5+
use setasign\Fpdi\FPDI;
6+
use Illuminate\Filesystem\Filesystem;
7+
use Illuminate\Support\Collection;
8+
9+
class PDFMerger {
10+
/**
11+
* Access the filesystem on an oop base
12+
*
13+
* @var Filesystem
14+
*/
15+
protected $filesystem = Filesystem::class;
16+
/**
17+
* Hold all the files which will be merged
18+
*
19+
* @var Collection
20+
*/
21+
protected $files = Collection::class;
22+
/**
23+
* Holds every tmp file so they can be removed during the deconstruction
24+
*
25+
* @var Collection
26+
*/
27+
protected $tmpFiles = Collection::class;
28+
/**
29+
* The actual PDF Service
30+
*
31+
* @var FPDI
32+
*/
33+
protected $fpdi = FPDI::class;
34+
/**
35+
* The final file name
36+
*
37+
* @var string
38+
*/
39+
protected $fileName = 'undefined.pdf';
40+
/**
41+
* Construct and initialize a new instance
42+
* @param Filesystem $Filesystem
43+
*/
44+
public function __construct(Filesystem $filesystem){
45+
$this->filesystem = $filesystem;
46+
$this->fpdi = new FPDI();
47+
$this->tmpFiles = collect([]);
48+
$this->files = collect([]);
49+
}
50+
/**
51+
* The class deconstructor method
52+
*/
53+
public function __destruct() {
54+
$filesystem = $this->filesystem;
55+
$this->tmpFiles->each(function($filePath) use($filesystem){
56+
$filesystem->delete($filePath);
57+
});
58+
}
59+
/**
60+
* Initialize a new internal instance of FPDI in order to prevent any problems with shared resources
61+
* Please visit https://www.setasign.com/products/fpdi/manual/#p-159 for more information on this issue
62+
*
63+
* @return self
64+
*/
65+
public function init(){
66+
return $this;
67+
}
68+
/**
69+
* Stream the merged PDF content
70+
*
71+
* @return string
72+
*/
73+
public function inline(){
74+
return $this->fpdi->Output($this->fileName, 'I');
75+
}
76+
/**
77+
* Download the merged PDF content
78+
*
79+
* @return string
80+
*/
81+
public function download(){
82+
return $this->fpdi->Output($this->fileName, 'D');
83+
}
84+
/**
85+
* Save the merged PDF content to the filesystem
86+
*
87+
* @return string
88+
*/
89+
public function save($filePath = null){
90+
return $this->filesystem->put($filePath ? $filePath : $this->fileName, $this->string());
91+
}
92+
/**
93+
* Get the merged PDF content as binary string
94+
*
95+
* @return string
96+
*/
97+
public function string(){
98+
return $this->fpdi->Output($this->fileName, 'S');
99+
}
100+
/**
101+
* Set the generated PDF fileName
102+
* @param string $fileName
103+
*
104+
* @return string
105+
*/
106+
public function setFileName($fileName){
107+
$this->fileName = $fileName;
108+
return $this;
109+
}
110+
/**
111+
* Add a PDF for inclusion in the merge with a binary string. Pages should be formatted: 1,3,6, 12-16.
112+
* @param string $string
113+
* @param mixed $pages
114+
* @param mixed $orientation
115+
*
116+
* @return void
117+
*/
118+
public function addPDFString($string, $pages = 'all', $orientation = null){
119+
$filePath = storage_path('tmp/'.str_random(16).'.pdf');
120+
$this->filesystem->put($filePath, $string);
121+
$this->tmpFiles->push($filePath);
122+
return $this->addPathToPDF($filePath, $pages, $orientation);
123+
}
124+
/**
125+
* Add a PDF for inclusion in the merge with a valid file path. Pages should be formatted: 1,3,6, 12-16.
126+
* @param string $filePath
127+
* @param string $pages
128+
* @param string $orientation
129+
*
130+
* @return self
131+
*
132+
* @throws \Exception if the given pages aren't correct
133+
*/
134+
public function addPathToPDF($filePath, $pages = 'all', $orientation = null) {
135+
if (file_exists($filePath)) {
136+
$filePath = $this->convertPDFVersion($filePath);
137+
if (!is_array($pages) && strtolower($pages) != 'all') {
138+
throw new \Exception($filePath."'s pages could not be validated");
139+
}
140+
$this->files->push([
141+
'name' => $filePath,
142+
'pages' => $pages,
143+
'orientation' => $orientation
144+
]);
145+
} else {
146+
throw new \Exception("Could not locate PDF on '$filePath'");
147+
}
148+
return $this;
149+
}
150+
/**
151+
* Merges your provided PDFs and outputs to specified location.
152+
* @param string $orientation
153+
*
154+
* @return void
155+
*
156+
* @throws \Exception if there are now PDFs to merge
157+
*/
158+
public function merge($orientation = 'P') {
159+
if ($this->files->count() == 0) {
160+
throw new \Exception("No PDFs to merge.");
161+
}
162+
$fpdi = $this->fpdi;
163+
$files = $this->files;
164+
foreach($files as $file){
165+
$file['orientation'] = is_null($file['orientation']) ? $orientation : $file['orientation'];
166+
$count = $fpdi->setSourceFile($file['name']);
167+
if($file['pages'] == 'all') {
168+
for ($i = 1; $i <= $count; $i++) {
169+
$template = $fpdi->importPage($i);
170+
$size = $fpdi->getTemplateSize($template);
171+
$fpdi->AddPage($file['orientation'], [$size['width'], $size['height']]);
172+
$fpdi->useTemplate($template);
173+
}
174+
}else {
175+
foreach ($file['pages'] as $page) {
176+
if (!$template = $fpdi->importPage($page)) {
177+
throw new \Exception("Could not load page '$page' in PDF '".$file['name']."'. Check that the page exists.");
178+
}
179+
$size = $fpdi->getTemplateSize($template);
180+
$fpdi->AddPage($file['orientation'], [$size['width'], $size['height']]);
181+
$fpdi->useTemplate($template);
182+
}
183+
}
184+
}
185+
}
186+
187+
/**
188+
* Converts PDF if version is above 1.4
189+
* @param string $filePath
190+
*
191+
* @return string
192+
*/
193+
protected function convertPDFVersion($filePath){
194+
$pdf = fopen($filePath, "r");
195+
$first_line = fgets($pdf);
196+
fclose($pdf);
197+
//extract version number
198+
preg_match_all('!\d+!', $first_line, $matches);
199+
$pdfversion = implode('.', $matches[0]);
200+
if($pdfversion > "1.4"){
201+
$newFilePath = storage_path('tmp/' . str_random(16) . '.pdf');
202+
//execute shell script that converts PDF to correct version and saves it to tmp folder
203+
shell_exec('gs -dBATCH -dNOPAUSE -q -sDEVICE=pdfwrite -sOutputFile="'. $newFilePath . '" "' . $filePath . '"');
204+
$this->tmpFiles->push($newFilePath);
205+
$filePath = $newFilePath;
206+
}
207+
//return correct file path
208+
return $filePath;
209+
}
210+
}

src/facades/PDFMergerFacade.php

+14
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,14 @@
1+
<?php
2+
3+
namespace GrofGraf\PDFMerger\Facades;
4+
5+
use \Illuminate\Support\Facades\Facade;
6+
7+
8+
class PDFMergerFacade extends Facade {
9+
10+
protected static function getFacadeAccessor() {
11+
return 'PDFMerger';
12+
}
13+
14+
}
+32
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,32 @@
1+
<?php
2+
3+
namespace GrofGraf\PDFMerger\Providers;
4+
5+
use Illuminate\Support\ServiceProvider;
6+
use GrofGraf\PDFMerger\PDFMerger;
7+
8+
class PDFMergerServiceProvider extends ServiceProvider
9+
{
10+
/**
11+
* Bootstrap services.
12+
*
13+
* @return void
14+
*/
15+
public function boot()
16+
{
17+
//
18+
}
19+
20+
/**
21+
* Register services.
22+
*
23+
* @return void
24+
*/
25+
public function register()
26+
{
27+
$this->app->singleton('PDFMerger', function ($app) {
28+
$pdfMerger = new PDFMerger($app['files']);
29+
return $pdfMerger;
30+
});
31+
}
32+
}

0 commit comments

Comments
 (0)