Proyectos de Subversion Moodle

Rev

| Ultima modificación | Ver Log |

Rev Autor Línea Nro. Línea
1441 ariadna 1
<?php
2
 
3
namespace PhpOffice\PhpSpreadsheet\Writer\Xlsx;
4
 
5
use PhpOffice\PhpSpreadsheet\Cell\Coordinate;
6
use PhpOffice\PhpSpreadsheet\Comment;
7
use PhpOffice\PhpSpreadsheet\Reader\Xlsx\Namespaces;
8
use PhpOffice\PhpSpreadsheet\Shared\XMLWriter;
9
use PhpOffice\PhpSpreadsheet\Style\Alignment;
10
 
11
class Comments extends WriterPart
12
{
13
    private const VALID_HORIZONTAL_ALIGNMENT = [
14
        Alignment::HORIZONTAL_CENTER,
15
        Alignment::HORIZONTAL_DISTRIBUTED,
16
        Alignment::HORIZONTAL_JUSTIFY,
17
        Alignment::HORIZONTAL_LEFT,
18
        Alignment::HORIZONTAL_RIGHT,
19
    ];
20
 
21
    /**
22
     * Write comments to XML format.
23
     *
24
     * @return string XML Output
25
     */
26
    public function writeComments(\PhpOffice\PhpSpreadsheet\Worksheet\Worksheet $worksheet): string
27
    {
28
        // Create XML writer
29
        $objWriter = null;
30
        if ($this->getParentWriter()->getUseDiskCaching()) {
31
            $objWriter = new XMLWriter(XMLWriter::STORAGE_DISK, $this->getParentWriter()->getDiskCachingDirectory());
32
        } else {
33
            $objWriter = new XMLWriter(XMLWriter::STORAGE_MEMORY);
34
        }
35
 
36
        // XML header
37
        $objWriter->startDocument('1.0', 'UTF-8', 'yes');
38
 
39
        // Comments cache
40
        $comments = $worksheet->getComments();
41
 
42
        // Authors cache
43
        $authors = [];
44
        $authorId = 0;
45
        foreach ($comments as $comment) {
46
            if (!isset($authors[$comment->getAuthor()])) {
47
                $authors[$comment->getAuthor()] = $authorId++;
48
            }
49
        }
50
 
51
        // comments
52
        $objWriter->startElement('comments');
53
        $objWriter->writeAttribute('xmlns', Namespaces::MAIN);
54
 
55
        // Loop through authors
56
        $objWriter->startElement('authors');
57
        foreach ($authors as $author => $index) {
58
            $objWriter->writeElement('author', $author);
59
        }
60
        $objWriter->endElement();
61
 
62
        // Loop through comments
63
        $objWriter->startElement('commentList');
64
        foreach ($comments as $key => $value) {
65
            $this->writeComment($objWriter, $key, $value, $authors);
66
        }
67
        $objWriter->endElement();
68
 
69
        $objWriter->endElement();
70
 
71
        // Return
72
        return $objWriter->getData();
73
    }
74
 
75
    /**
76
     * Write comment to XML format.
77
     *
78
     * @param string $cellReference Cell reference
79
     * @param Comment $comment Comment
80
     * @param array $authors Array of authors
81
     */
82
    private function writeComment(XMLWriter $objWriter, string $cellReference, Comment $comment, array $authors): void
83
    {
84
        // comment
85
        $objWriter->startElement('comment');
86
        $objWriter->writeAttribute('ref', $cellReference);
87
        $objWriter->writeAttribute('authorId', $authors[$comment->getAuthor()]);
88
 
89
        // text
90
        $objWriter->startElement('text');
91
        $this->getParentWriter()->getWriterPartstringtable()->writeRichText($objWriter, $comment->getText());
92
        $objWriter->endElement();
93
 
94
        $objWriter->endElement();
95
    }
96
 
97
    /**
98
     * Write VML comments to XML format.
99
     *
100
     * @return string XML Output
101
     */
102
    public function writeVMLComments(\PhpOffice\PhpSpreadsheet\Worksheet\Worksheet $worksheet): string
103
    {
104
        // Create XML writer
105
        $objWriter = null;
106
        if ($this->getParentWriter()->getUseDiskCaching()) {
107
            $objWriter = new XMLWriter(XMLWriter::STORAGE_DISK, $this->getParentWriter()->getDiskCachingDirectory());
108
        } else {
109
            $objWriter = new XMLWriter(XMLWriter::STORAGE_MEMORY);
110
        }
111
 
112
        // XML header
113
        $objWriter->startDocument('1.0', 'UTF-8', 'yes');
114
 
115
        // Comments cache
116
        $comments = $worksheet->getComments();
117
 
118
        // xml
119
        $objWriter->startElement('xml');
120
        $objWriter->writeAttribute('xmlns:v', Namespaces::URN_VML);
121
        $objWriter->writeAttribute('xmlns:o', Namespaces::URN_MSOFFICE);
122
        $objWriter->writeAttribute('xmlns:x', Namespaces::URN_EXCEL);
123
 
124
        // o:shapelayout
125
        $objWriter->startElement('o:shapelayout');
126
        $objWriter->writeAttribute('v:ext', 'edit');
127
 
128
        // o:idmap
129
        $objWriter->startElement('o:idmap');
130
        $objWriter->writeAttribute('v:ext', 'edit');
131
        $objWriter->writeAttribute('data', '1');
132
        $objWriter->endElement();
133
 
134
        $objWriter->endElement();
135
 
136
        // v:shapetype
137
        $objWriter->startElement('v:shapetype');
138
        $objWriter->writeAttribute('id', '_x0000_t202');
139
        $objWriter->writeAttribute('coordsize', '21600,21600');
140
        $objWriter->writeAttribute('o:spt', '202');
141
        $objWriter->writeAttribute('path', 'm,l,21600r21600,l21600,xe');
142
 
143
        // v:stroke
144
        $objWriter->startElement('v:stroke');
145
        $objWriter->writeAttribute('joinstyle', 'miter');
146
        $objWriter->endElement();
147
 
148
        // v:path
149
        $objWriter->startElement('v:path');
150
        $objWriter->writeAttribute('gradientshapeok', 't');
151
        $objWriter->writeAttribute('o:connecttype', 'rect');
152
        $objWriter->endElement();
153
 
154
        $objWriter->endElement();
155
 
156
        // Loop through comments
157
        foreach ($comments as $key => $value) {
158
            $this->writeVMLComment($objWriter, $key, $value);
159
        }
160
 
161
        $objWriter->endElement();
162
 
163
        // Return
164
        return $objWriter->getData();
165
    }
166
 
167
    /**
168
     * Write VML comment to XML format.
169
     *
170
     * @param string $cellReference Cell reference, eg: 'A1'
171
     * @param Comment $comment Comment
172
     */
173
    private function writeVMLComment(XMLWriter $objWriter, string $cellReference, Comment $comment): void
174
    {
175
        // Metadata
176
        [$column, $row] = Coordinate::indexesFromString($cellReference);
177
        $id = 1024 + $column + $row;
178
        $id = substr("$id", 0, 4);
179
 
180
        // v:shape
181
        $objWriter->startElement('v:shape');
182
        $objWriter->writeAttribute('id', '_x0000_s' . $id);
183
        $objWriter->writeAttribute('type', '#_x0000_t202');
184
        $objWriter->writeAttribute('style', 'position:absolute;margin-left:' . $comment->getMarginLeft() . ';margin-top:' . $comment->getMarginTop() . ';width:' . $comment->getWidth() . ';height:' . $comment->getHeight() . ';z-index:1;visibility:' . ($comment->getVisible() ? 'visible' : 'hidden'));
185
        $objWriter->writeAttribute('fillcolor', '#' . $comment->getFillColor()->getRGB());
186
        $objWriter->writeAttribute('o:insetmode', 'auto');
187
 
188
        // v:fill
189
        $objWriter->startElement('v:fill');
190
        $objWriter->writeAttribute('color2', '#' . $comment->getFillColor()->getRGB());
191
        if ($comment->hasBackgroundImage()) {
192
            $bgImage = $comment->getBackgroundImage();
193
            $objWriter->writeAttribute('o:relid', 'rId' . $bgImage->getImageIndex());
194
            $objWriter->writeAttribute('o:title', $bgImage->getName());
195
            $objWriter->writeAttribute('type', 'frame');
196
        }
197
        $objWriter->endElement();
198
 
199
        // v:shadow
200
        $objWriter->startElement('v:shadow');
201
        $objWriter->writeAttribute('on', 't');
202
        $objWriter->writeAttribute('color', 'black');
203
        $objWriter->writeAttribute('obscured', 't');
204
        $objWriter->endElement();
205
 
206
        // v:path
207
        $objWriter->startElement('v:path');
208
        $objWriter->writeAttribute('o:connecttype', 'none');
209
        $objWriter->endElement();
210
 
211
        // v:textbox
212
        $textBoxArray = [Comment::TEXTBOX_DIRECTION_RTL => 'rtl', Comment::TEXTBOX_DIRECTION_LTR => 'ltr'];
213
        $textboxRtl = $textBoxArray[strtolower($comment->getTextBoxDirection())] ?? 'auto';
214
        $objWriter->startElement('v:textbox');
215
        $objWriter->writeAttribute('style', "mso-direction-alt:$textboxRtl");
216
 
217
        // div
218
        $objWriter->startElement('div');
219
        $objWriter->writeAttribute('style', ($textboxRtl === 'rtl' ? 'text-align:right;direction:rtl' : 'text-align:left'));
220
        $objWriter->endElement();
221
 
222
        $objWriter->endElement();
223
 
224
        // x:ClientData
225
        $objWriter->startElement('x:ClientData');
226
        $objWriter->writeAttribute('ObjectType', 'Note');
227
 
228
        // x:MoveWithCells
229
        $objWriter->writeElement('x:MoveWithCells', '');
230
 
231
        // x:SizeWithCells
232
        $objWriter->writeElement('x:SizeWithCells', '');
233
 
234
        // x:AutoFill
235
        $objWriter->writeElement('x:AutoFill', 'False');
236
 
237
        // x:TextHAlign horizontal alignment of text
238
        $alignment = strtolower($comment->getAlignment());
239
        if (in_array($alignment, self::VALID_HORIZONTAL_ALIGNMENT, true)) {
240
            $objWriter->writeElement('x:TextHAlign', ucfirst($alignment));
241
        }
242
 
243
        // x:Row
244
        $objWriter->writeElement('x:Row', (string) ($row - 1));
245
 
246
        // x:Column
247
        $objWriter->writeElement('x:Column', (string) ($column - 1));
248
 
249
        $objWriter->endElement();
250
 
251
        $objWriter->endElement();
252
    }
253
}