diff options
Diffstat (limited to 'linden/indra/newview/llpreviewtexture.cpp')
-rw-r--r-- | linden/indra/newview/llpreviewtexture.cpp | 387 |
1 files changed, 387 insertions, 0 deletions
diff --git a/linden/indra/newview/llpreviewtexture.cpp b/linden/indra/newview/llpreviewtexture.cpp new file mode 100644 index 0000000..7948db4 --- /dev/null +++ b/linden/indra/newview/llpreviewtexture.cpp | |||
@@ -0,0 +1,387 @@ | |||
1 | /** | ||
2 | * @file llpreviewtexture.cpp | ||
3 | * @brief LLPreviewTexture class implementation | ||
4 | * | ||
5 | * Copyright (c) 2002-2007, Linden Research, Inc. | ||
6 | * | ||
7 | * The source code in this file ("Source Code") is provided by Linden Lab | ||
8 | * to you under the terms of the GNU General Public License, version 2.0 | ||
9 | * ("GPL"), unless you have obtained a separate licensing agreement | ||
10 | * ("Other License"), formally executed by you and Linden Lab. Terms of | ||
11 | * the GPL can be found in doc/GPL-license.txt in this distribution, or | ||
12 | * online at http://secondlife.com/developers/opensource/gplv2 | ||
13 | * | ||
14 | * There are special exceptions to the terms and conditions of the GPL as | ||
15 | * it is applied to this Source Code. View the full text of the exception | ||
16 | * in the file doc/FLOSS-exception.txt in this software distribution, or | ||
17 | * online at http://secondlife.com/developers/opensource/flossexception | ||
18 | * | ||
19 | * By copying, modifying or distributing this software, you acknowledge | ||
20 | * that you have read and understood your obligations described above, | ||
21 | * and agree to abide by those obligations. | ||
22 | * | ||
23 | * ALL LINDEN LAB SOURCE CODE IS PROVIDED "AS IS." LINDEN LAB MAKES NO | ||
24 | * WARRANTIES, EXPRESS, IMPLIED OR OTHERWISE, REGARDING ITS ACCURACY, | ||
25 | * COMPLETENESS OR PERFORMANCE. | ||
26 | */ | ||
27 | |||
28 | #include "llviewerprecompiledheaders.h" | ||
29 | |||
30 | #include "llpreviewtexture.h" | ||
31 | |||
32 | #include "llviewerimage.h" | ||
33 | #include "llviewerimagelist.h" | ||
34 | #include "llresmgr.h" | ||
35 | #include "llagent.h" | ||
36 | #include "llbutton.h" | ||
37 | #include "llui.h" | ||
38 | #include "llinventoryview.h" | ||
39 | #include "llinventory.h" | ||
40 | #include "llviewerwindow.h" | ||
41 | #include "lltextbox.h" | ||
42 | #include "llimagetga.h" | ||
43 | #include "llfilepicker.h" | ||
44 | #include "llvieweruictrlfactory.h" | ||
45 | |||
46 | const S32 PREVIEW_TEXTURE_MIN_WIDTH = 300; | ||
47 | const S32 PREVIEW_TEXTURE_MIN_HEIGHT = 120; | ||
48 | |||
49 | const S32 CLIENT_RECT_VPAD = 4; | ||
50 | |||
51 | const F32 SECONDS_TO_SHOW_FILE_SAVED_MSG = 8.f; | ||
52 | |||
53 | LLPreviewTexture::LLPreviewTexture(const std::string& name, | ||
54 | const LLRect& rect, | ||
55 | const std::string& title, | ||
56 | const LLUUID& item_uuid, | ||
57 | const LLUUID& object_id, | ||
58 | BOOL show_keep_discard) | ||
59 | : LLPreview(name, rect, title, item_uuid, object_id, TRUE, PREVIEW_TEXTURE_MIN_WIDTH, PREVIEW_TEXTURE_MIN_HEIGHT ), | ||
60 | mLoadingFullImage( FALSE ), | ||
61 | mShowKeepDiscard(show_keep_discard), | ||
62 | mCopyToInv(FALSE), | ||
63 | mIsCopyable(FALSE), | ||
64 | mLastHeight(0), | ||
65 | mLastWidth(0) | ||
66 | { | ||
67 | LLInventoryItem *item = getItem(); | ||
68 | if(item) | ||
69 | { | ||
70 | mImageID = item->getAssetUUID(); | ||
71 | const LLPermissions& perm = item->getPermissions(); | ||
72 | U32 mask = PERM_NONE; | ||
73 | if(perm.getOwner() == gAgent.getID()) | ||
74 | { | ||
75 | mask = perm.getMaskBase(); | ||
76 | } | ||
77 | else if(gAgent.isInGroup(perm.getGroup())) | ||
78 | { | ||
79 | mask = perm.getMaskGroup(); | ||
80 | } | ||
81 | else | ||
82 | { | ||
83 | mask = perm.getMaskEveryone(); | ||
84 | } | ||
85 | if((mask & PERM_ITEM_UNRESTRICTED) == PERM_ITEM_UNRESTRICTED) | ||
86 | { | ||
87 | mIsCopyable = TRUE; | ||
88 | } | ||
89 | } | ||
90 | |||
91 | init(); | ||
92 | |||
93 | setTitle(title); | ||
94 | |||
95 | if (!getHost()) | ||
96 | { | ||
97 | LLRect curRect = getRect(); | ||
98 | translate(rect.mLeft - curRect.mLeft, rect.mTop - curRect.mTop); | ||
99 | } | ||
100 | } | ||
101 | |||
102 | |||
103 | // Note: uses asset_id as a dummy item id. | ||
104 | LLPreviewTexture::LLPreviewTexture( | ||
105 | const std::string& name, | ||
106 | const LLRect& rect, | ||
107 | const std::string& title, | ||
108 | const LLUUID& asset_id, | ||
109 | BOOL copy_to_inv) | ||
110 | : | ||
111 | LLPreview( | ||
112 | name, | ||
113 | rect, | ||
114 | title, | ||
115 | asset_id, | ||
116 | LLUUID::null, | ||
117 | TRUE, | ||
118 | PREVIEW_TEXTURE_MIN_WIDTH, | ||
119 | PREVIEW_TEXTURE_MIN_HEIGHT ), | ||
120 | mImageID(asset_id), | ||
121 | mLoadingFullImage( FALSE ), | ||
122 | mShowKeepDiscard(FALSE), | ||
123 | mCopyToInv(copy_to_inv), | ||
124 | mIsCopyable(TRUE), | ||
125 | mLastHeight(0), | ||
126 | mLastWidth(0) | ||
127 | { | ||
128 | |||
129 | init(); | ||
130 | |||
131 | setTitle(title); | ||
132 | |||
133 | LLRect curRect = getRect(); | ||
134 | translate(curRect.mLeft - rect.mLeft, curRect.mTop - rect.mTop); | ||
135 | |||
136 | } | ||
137 | |||
138 | |||
139 | LLPreviewTexture::~LLPreviewTexture() | ||
140 | { | ||
141 | if( mLoadingFullImage ) | ||
142 | { | ||
143 | getWindow()->decBusyCount(); | ||
144 | } | ||
145 | |||
146 | mImage = NULL; | ||
147 | } | ||
148 | |||
149 | |||
150 | void LLPreviewTexture::init() | ||
151 | { | ||
152 | |||
153 | |||
154 | if (mCopyToInv) | ||
155 | { | ||
156 | gUICtrlFactory->buildFloater(this,"floater_preview_embedded_texture.xml"); | ||
157 | |||
158 | childSetAction("Copy To Inventory",LLPreview::onBtnCopyToInv,this); | ||
159 | } | ||
160 | |||
161 | else if (mShowKeepDiscard) | ||
162 | { | ||
163 | gUICtrlFactory->buildFloater(this,"floater_preview_texture_keep_discard.xml"); | ||
164 | |||
165 | childSetAction("Keep",onKeepBtn,this); | ||
166 | childSetAction("Discard",onDiscardBtn,this); | ||
167 | } | ||
168 | |||
169 | else | ||
170 | { | ||
171 | gUICtrlFactory->buildFloater(this,"floater_preview_texture.xml"); | ||
172 | } | ||
173 | |||
174 | |||
175 | if (!mCopyToInv) | ||
176 | { | ||
177 | LLInventoryItem* item = getItem(); | ||
178 | |||
179 | if (item) | ||
180 | { | ||
181 | childSetCommitCallback("desc", LLPreview::onText, this); | ||
182 | childSetText("desc", item->getDescription()); | ||
183 | childSetPrevalidate("desc", &LLLineEditor::prevalidatePrintableNotPipe); | ||
184 | } | ||
185 | } | ||
186 | } | ||
187 | |||
188 | |||
189 | // virtual | ||
190 | BOOL LLPreviewTexture::canSaveAs() | ||
191 | { | ||
192 | return mIsCopyable && !mLoadingFullImage && mImage.notNull() && !mImage->isMissingAsset(); | ||
193 | } | ||
194 | |||
195 | |||
196 | // virtual | ||
197 | void LLPreviewTexture::saveAs() | ||
198 | { | ||
199 | if( !mLoadingFullImage ) | ||
200 | { | ||
201 | LLFilePicker& file_picker = LLFilePicker::instance(); | ||
202 | if( !file_picker.getSaveFile( LLFilePicker::FFSAVE_TGA ) ) | ||
203 | { | ||
204 | // User canceled save. | ||
205 | return; | ||
206 | } | ||
207 | mSaveFileName = file_picker.getFirstFile(); | ||
208 | mLoadingFullImage = TRUE; | ||
209 | getWindow()->incBusyCount(); | ||
210 | mImage->setLoadedCallback( LLPreviewTexture::onFileLoadedForSave, | ||
211 | 0, | ||
212 | TRUE, | ||
213 | new LLUUID( mItemUUID ) ); | ||
214 | } | ||
215 | } | ||
216 | |||
217 | |||
218 | // static | ||
219 | void LLPreviewTexture::onFileLoadedForSave(BOOL success, | ||
220 | LLViewerImage *src_vi, | ||
221 | LLImageRaw* src, | ||
222 | LLImageRaw* aux_src, | ||
223 | S32 discard_level, | ||
224 | BOOL final, | ||
225 | void* userdata) | ||
226 | { | ||
227 | LLUUID* item_uuid = (LLUUID*) userdata; | ||
228 | LLPreviewTexture* self = NULL; | ||
229 | preview_map_t::iterator found_it = LLPreview::sInstances.find(*item_uuid); | ||
230 | if(found_it != LLPreview::sInstances.end()) | ||
231 | { | ||
232 | self = (LLPreviewTexture*) found_it->second; | ||
233 | } | ||
234 | |||
235 | if( final || !success ) | ||
236 | { | ||
237 | delete item_uuid; | ||
238 | |||
239 | if( self ) | ||
240 | { | ||
241 | self->getWindow()->decBusyCount(); | ||
242 | self->mLoadingFullImage = FALSE; | ||
243 | } | ||
244 | } | ||
245 | |||
246 | if( self && final && success ) | ||
247 | { | ||
248 | LLPointer<LLImageTGA> image_tga = new LLImageTGA; | ||
249 | if( !image_tga->encode( src ) ) | ||
250 | { | ||
251 | LLStringBase<char>::format_map_t args; | ||
252 | args["[FILE]"] = self->mSaveFileName; | ||
253 | gViewerWindow->alertXml("CannotEncodeFile", args); | ||
254 | } | ||
255 | else if( !image_tga->save( self->mSaveFileName ) ) | ||
256 | { | ||
257 | LLStringBase<char>::format_map_t args; | ||
258 | args["[FILE]"] = self->mSaveFileName; | ||
259 | gViewerWindow->alertXml("CannotWriteFile", args); | ||
260 | } | ||
261 | else | ||
262 | { | ||
263 | self->mSavedFileTimer.reset(); | ||
264 | self->mSavedFileTimer.setTimerExpirySec( SECONDS_TO_SHOW_FILE_SAVED_MSG ); | ||
265 | } | ||
266 | |||
267 | self->mSaveFileName.clear(); | ||
268 | } | ||
269 | |||
270 | if( self && !success ) | ||
271 | { | ||
272 | gViewerWindow->alertXml("CannotDownloadFile"); | ||
273 | } | ||
274 | } | ||
275 | |||
276 | |||
277 | // It takes a while until we get height and width information. | ||
278 | // When we receive it, reshape the window accordingly. | ||
279 | void LLPreviewTexture::updateAspectRatio() | ||
280 | { | ||
281 | if (!mImage) return; | ||
282 | |||
283 | S32 image_height = llmax(1, mImage->getHeight(0)); | ||
284 | S32 image_width = llmax(1, mImage->getWidth(0)); | ||
285 | // Attempt to make the image 1:1 on screen. | ||
286 | // If that fails, cut width by half. | ||
287 | S32 client_width = image_width; | ||
288 | S32 horiz_pad = 2 * (LLPANEL_BORDER_WIDTH + PREVIEW_PAD) + PREVIEW_RESIZE_HANDLE_SIZE; | ||
289 | S32 screen_width = gViewerWindow->getWindowWidth(); | ||
290 | S32 max_client_width = screen_width - horiz_pad; | ||
291 | |||
292 | while (client_width > max_client_width) | ||
293 | { | ||
294 | client_width /= 2; | ||
295 | } | ||
296 | |||
297 | // Demand width at least 128 | ||
298 | if (client_width < 128) | ||
299 | { | ||
300 | client_width = 128; | ||
301 | } | ||
302 | |||
303 | S32 view_width = client_width + horiz_pad; | ||
304 | |||
305 | // Adjust the height based on the width computed above. | ||
306 | F32 inv_aspect_ratio = (F32) image_height / (F32) image_width; | ||
307 | S32 client_height = llround(client_width * inv_aspect_ratio); | ||
308 | S32 view_height = | ||
309 | PREVIEW_HEADER_SIZE + // header (includes top border) | ||
310 | client_height + 2 * CLIENT_RECT_VPAD + // texture plus uniform spacing (which leaves room for resize handle) | ||
311 | LLPANEL_BORDER_WIDTH; // bottom border | ||
312 | |||
313 | // set text on dimensions display (should be moved out of here and into a callback of some sort) | ||
314 | childSetTextArg("dimensions", "[WIDTH]", llformat("%d", mImage->mFullWidth)); | ||
315 | childSetTextArg("dimensions", "[HEIGHT]", llformat("%d", mImage->mFullHeight)); | ||
316 | |||
317 | // add space for dimensions | ||
318 | S32 info_height = 0; | ||
319 | LLRect dim_rect; | ||
320 | childGetRect("dimensions", dim_rect); | ||
321 | S32 dim_height = dim_rect.getHeight(); | ||
322 | info_height += dim_height + CLIENT_RECT_VPAD; | ||
323 | view_height += info_height; | ||
324 | |||
325 | S32 button_height = 0; | ||
326 | if (mShowKeepDiscard || mCopyToInv) { //mCopyToInvBtn | ||
327 | |||
328 | // add space for buttons | ||
329 | view_height += BTN_HEIGHT + CLIENT_RECT_VPAD; | ||
330 | button_height = BTN_HEIGHT + PREVIEW_PAD; | ||
331 | } | ||
332 | |||
333 | LLRect window_rect(0, mRect.getHeight(), mRect.getWidth(), 0); | ||
334 | window_rect.mTop -= (PREVIEW_HEADER_SIZE + CLIENT_RECT_VPAD); | ||
335 | window_rect.mBottom += PREVIEW_BORDER + button_height + CLIENT_RECT_VPAD + info_height + CLIENT_RECT_VPAD; | ||
336 | LLMultiFloater* hostp = getHost(); | ||
337 | if (hostp) | ||
338 | { | ||
339 | // try to keep aspect ratio when hosted, as hosting view can resize without user input | ||
340 | mClientRect.setLeftTopAndSize(window_rect.getCenterX() - (client_width / 2), window_rect.mTop, client_width, client_height); | ||
341 | } | ||
342 | else | ||
343 | { | ||
344 | mClientRect.setLeftTopAndSize(LLPANEL_BORDER_WIDTH + PREVIEW_PAD + 6, | ||
345 | mRect.getHeight() - (PREVIEW_HEADER_SIZE + CLIENT_RECT_VPAD), | ||
346 | mRect.getWidth() - horiz_pad, | ||
347 | mRect.getHeight() - (view_height - client_height) - 8); | ||
348 | } | ||
349 | |||
350 | if (mImage->mFullHeight > mLastHeight && mImage->mFullWidth > mLastWidth) | ||
351 | { | ||
352 | mLastWidth = image_width; | ||
353 | mLastHeight = image_height; | ||
354 | |||
355 | S32 old_top = mRect.mTop; | ||
356 | S32 old_left = mRect.mLeft; | ||
357 | if (hostp) | ||
358 | { | ||
359 | hostp->growToFit(this, view_width, view_height); | ||
360 | } | ||
361 | else | ||
362 | { | ||
363 | reshape( view_width, view_height ); | ||
364 | S32 new_bottom = old_top - mRect.getHeight(); | ||
365 | setOrigin( old_left, new_bottom ); | ||
366 | } | ||
367 | |||
368 | // Try to keep whole view onscreen, don't allow partial offscreen. | ||
369 | gFloaterView->adjustToFitScreen(this, FALSE); | ||
370 | } | ||
371 | } | ||
372 | |||
373 | void LLPreviewTexture::loadAsset() | ||
374 | { | ||
375 | mImage = gImageList.getImage(mImageID, MIPMAP_FALSE, FALSE); | ||
376 | mImage->setBoostLevel(LLViewerImage::BOOST_PREVIEW); | ||
377 | mAssetStatus = PREVIEW_ASSET_LOADING; | ||
378 | } | ||
379 | |||
380 | LLPreview::EAssetStatus LLPreviewTexture::getAssetStatus() | ||
381 | { | ||
382 | if (mImage.notNull() && (mImage->mFullWidth * mImage->mFullHeight > 0)) | ||
383 | { | ||
384 | mAssetStatus = PREVIEW_ASSET_LOADED; | ||
385 | } | ||
386 | return mAssetStatus; | ||
387 | } | ||