diff options
| author | Merov Linden <merov@lindenlab.com> | 2010-12-22 14:06:46 -0800 | 
|---|---|---|
| committer | Merov Linden <merov@lindenlab.com> | 2010-12-22 14:06:46 -0800 | 
| commit | c86d6a7bbb7c0db7665b76cf52b12b90c6e98c6d (patch) | |
| tree | 1056d6dc941e2aed458b459901fbb448c6e40ef7 | |
| parent | ab100825bd0064dc64b9ef8bea1c70bc04090716 (diff) | |
| parent | 6182ded4e9b4ba076add7a840c535589d36ec283 (diff) | |
STORM-808 : merge webkit upgrade to Qt v4.7.1
49 files changed, 2207 insertions, 693 deletions
| diff --git a/indra/llplugin/llpluginclassmedia.cpp b/indra/llplugin/llpluginclassmedia.cpp index 69ed0fb09c..595c470a19 100644 --- a/indra/llplugin/llpluginclassmedia.cpp +++ b/indra/llplugin/llpluginclassmedia.cpp @@ -160,7 +160,7 @@ void LLPluginClassMedia::idle(void)  		mPlugin->idle();  	} -	if((mMediaWidth == -1) || (!mTextureParamsReceived) || (mPlugin == NULL) || (mPlugin->isBlocked())) +	if((mMediaWidth == -1) || (!mTextureParamsReceived) || (mPlugin == NULL) || (mPlugin->isBlocked()) || (mOwner == NULL))  	{  		// Can't process a size change at this time  	} @@ -522,7 +522,15 @@ bool LLPluginClassMedia::keyEvent(EKeyEventType type, int key_code, MASK modifie  			}  		break;  	} -	 + +#if LL_DARWIN	 +	if(modifiers & MASK_ALT) +	{ +		// Option-key modified characters should be handled by the unicode input path instead of this one. +		result = false; +	} +#endif +  	if(result)  	{  		LLPluginMessage message(LLPLUGIN_MESSAGE_CLASS_MEDIA, "key_event"); @@ -674,7 +682,21 @@ void LLPluginClassMedia::sendPickFileResponse(const std::string &file)  {  	LLPluginMessage message(LLPLUGIN_MESSAGE_CLASS_MEDIA, "pick_file_response");  	message.setValue("file", file); -	if(mPlugin->isBlocked()) +	if(mPlugin && mPlugin->isBlocked()) +	{ +		// If the plugin sent a blocking pick-file request, the response should unblock it. +		message.setValueBoolean("blocking_response", true); +	} +	sendMessage(message); +} + +void LLPluginClassMedia::sendAuthResponse(bool ok, const std::string &username, const std::string &password) +{ +	LLPluginMessage message(LLPLUGIN_MESSAGE_CLASS_MEDIA, "auth_response"); +	message.setValueBoolean("ok", ok); +	message.setValue("username", username); +	message.setValue("password", password); +	if(mPlugin && mPlugin->isBlocked())  	{  		// If the plugin sent a blocking pick-file request, the response should unblock it.  		message.setValueBoolean("blocking_response", true); @@ -947,6 +969,12 @@ void LLPluginClassMedia::receivePluginMessage(const LLPluginMessage &message)  		{  			mediaEvent(LLPluginClassMediaOwner::MEDIA_EVENT_PICK_FILE_REQUEST);  		} +		else if(message_name == "auth_request") +		{ +			mAuthURL = message.getValue("url"); +			mAuthRealm = message.getValue("realm"); +			mediaEvent(LLPluginClassMediaOwner::MEDIA_EVENT_AUTH_REQUEST); +		}  		else  		{  			LL_WARNS("Plugin") << "Unknown " << message_name << " class message: " << message_name << LL_ENDL; @@ -1019,6 +1047,15 @@ void LLPluginClassMedia::receivePluginMessage(const LLPluginMessage &message)  			mediaEvent(LLPluginClassMediaOwner::MEDIA_EVENT_GEOMETRY_CHANGE);  		} +		else if(message_name == "link_hovered") +		{ +			// text is not currently used -- the tooltip hover text is taken from the "title". +			mHoverLink = message.getValue("link"); +			mHoverText = message.getValue("title"); +			// message.getValue("text"); +				 +			mediaEvent(LLPluginClassMediaOwner::MEDIA_EVENT_LINK_HOVERED); +		}  		else  		{  			LL_WARNS("Plugin") << "Unknown " << message_name << " class message: " << message_name << LL_ENDL; @@ -1192,6 +1229,20 @@ void LLPluginClassMedia::proxyWindowClosed(const std::string &uuid)  	sendMessage(message);  } +void LLPluginClassMedia::ignore_ssl_cert_errors(bool ignore) +{ +	LLPluginMessage message(LLPLUGIN_MESSAGE_CLASS_MEDIA_BROWSER, "ignore_ssl_cert_errors"); +	message.setValueBoolean("ignore", ignore); +	sendMessage(message); +} + +void LLPluginClassMedia::addCertificateFilePath(const std::string& path) +{ +	LLPluginMessage message(LLPLUGIN_MESSAGE_CLASS_MEDIA_BROWSER, "add_certificate_file_path"); +	message.setValue("path", path); +	sendMessage(message); +} +  void LLPluginClassMedia::crashPlugin()  {  	LLPluginMessage message(LLPLUGIN_MESSAGE_CLASS_INTERNAL, "crash"); diff --git a/indra/llplugin/llpluginclassmedia.h b/indra/llplugin/llpluginclassmedia.h index 9cb67fe909..c826e13c40 100644 --- a/indra/llplugin/llpluginclassmedia.h +++ b/indra/llplugin/llpluginclassmedia.h @@ -85,6 +85,8 @@ public:  	void setBackgroundColor(LLColor4 color) { mBackgroundColor = color; }; +	void setOwner(LLPluginClassMediaOwner *owner) { mOwner = owner; }; +	  	// Returns true if all of the texture parameters (depth, format, size, and texture size) are set up and consistent.  	// This will initially be false, and will also be false for some time after setSize while the resize is processed.  	// Note that if this returns true, it is safe to use all the get() functions above without checking for invalid return values @@ -159,6 +161,8 @@ public:  	void sendPickFileResponse(const std::string &file); +	void sendAuthResponse(bool ok, const std::string &username, const std::string &password); +  	// Valid after a MEDIA_EVENT_CURSOR_CHANGED event  	std::string getCursorName() const { return mCursorName; }; @@ -198,6 +202,8 @@ public:  	void setBrowserUserAgent(const std::string& user_agent);  	void proxyWindowOpened(const std::string &target, const std::string &uuid);  	void proxyWindowClosed(const std::string &uuid); +	void ignore_ssl_cert_errors(bool ignore); +	void addCertificateFilePath(const std::string& path);  	// This is valid after MEDIA_EVENT_NAVIGATE_BEGIN or MEDIA_EVENT_NAVIGATE_COMPLETE  	std::string	getNavigateURI() const { return mNavigateURI; }; @@ -231,7 +237,15 @@ public:  	S32 getGeometryY() const { return mGeometryY; };  	S32 getGeometryWidth() const { return mGeometryWidth; };  	S32 getGeometryHeight() const { return mGeometryHeight; }; +	 +	// These are valid during MEDIA_EVENT_AUTH_REQUEST +	std::string	getAuthURL() const { return mAuthURL; }; +	std::string	getAuthRealm() const { return mAuthRealm; }; +	// These are valid during MEDIA_EVENT_LINK_HOVERED +	std::string	getHoverText() const { return mHoverText; }; +	std::string	getHoverLink() const { return mHoverLink; }; +	  	std::string getMediaName() const { return mMediaName; };  	std::string getMediaDescription() const { return mMediaDescription; }; @@ -369,6 +383,10 @@ protected:  	S32				mGeometryY;  	S32				mGeometryWidth;  	S32				mGeometryHeight; +	std::string		mAuthURL; +	std::string		mAuthRealm; +	std::string		mHoverText; +	std::string		mHoverLink;  	/////////////////////////////////////////  	// media_time class diff --git a/indra/llplugin/llpluginclassmediaowner.h b/indra/llplugin/llpluginclassmediaowner.h index c9efff216c..42e93cc6d7 100644 --- a/indra/llplugin/llpluginclassmediaowner.h +++ b/indra/llplugin/llpluginclassmediaowner.h @@ -59,7 +59,11 @@ public:  		MEDIA_EVENT_GEOMETRY_CHANGE,		// The plugin requested its window geometry be changed (per the javascript window interface)  		MEDIA_EVENT_PLUGIN_FAILED_LAUNCH,	// The plugin failed to launch  -		MEDIA_EVENT_PLUGIN_FAILED			// The plugin died unexpectedly +		MEDIA_EVENT_PLUGIN_FAILED,			// The plugin died unexpectedly + +		MEDIA_EVENT_AUTH_REQUEST,			// The plugin wants to display an auth dialog + +		MEDIA_EVENT_LINK_HOVERED			// Got a "link hovered" event from the plugin  	} EMediaEvent; diff --git a/indra/llui/CMakeLists.txt b/indra/llui/CMakeLists.txt index 864f3f699e..33ab2e93b5 100644 --- a/indra/llui/CMakeLists.txt +++ b/indra/llui/CMakeLists.txt @@ -111,6 +111,7 @@ set(llui_SOURCE_FILES      llviewmodel.cpp      llview.cpp      llviewquery.cpp +    llwindowshade.cpp      )  set(llui_HEADER_FILES @@ -210,6 +211,7 @@ set(llui_HEADER_FILES      llviewmodel.h      llview.h      llviewquery.h +    llwindowshade.h      )  set_source_files_properties(${llui_HEADER_FILES} diff --git a/indra/llui/lllayoutstack.cpp b/indra/llui/lllayoutstack.cpp index ac30fce392..19ac4c58a8 100644 --- a/indra/llui/lllayoutstack.cpp +++ b/indra/llui/lllayoutstack.cpp @@ -38,6 +38,12 @@  static LLDefaultChildRegistry::Register<LLLayoutStack> register_layout_stack("layout_stack");  static LLLayoutStack::LayoutStackRegistry::Register<LLLayoutPanel> register_layout_panel("layout_panel"); +void LLLayoutStack::OrientationNames::declareValues() +{ +	declare("horizontal", HORIZONTAL); +	declare("vertical", VERTICAL); +} +  //  // LLLayoutPanel  // @@ -47,47 +53,47 @@ LLLayoutPanel::LLLayoutPanel(const Params& p)   	mMaxDim(p.max_dim),    	mAutoResize(p.auto_resize),   	mUserResize(p.user_resize), -		mCollapsed(FALSE), -		mCollapseAmt(0.f), -		mVisibleAmt(1.f), // default to fully visible -		mResizeBar(NULL)  -	{ +	mCollapsed(FALSE), +	mCollapseAmt(0.f), +	mVisibleAmt(1.f), // default to fully visible +	mResizeBar(NULL)  +{  	// panels initialized as hidden should not start out partially visible  	if (!getVisible()) -		{ +	{  		mVisibleAmt = 0.f; -		} -		} +	} +}  void LLLayoutPanel::initFromParams(const Params& p) -		{ +{  	LLPanel::initFromParams(p);  	setFollowsNone(); -	} +}  LLLayoutPanel::~LLLayoutPanel() -	{ -		// probably not necessary, but... -		delete mResizeBar; -		mResizeBar = NULL; -	} +{ +	// probably not necessary, but... +	delete mResizeBar; +	mResizeBar = NULL; +}  F32 LLLayoutPanel::getCollapseFactor(LLLayoutStack::ELayoutOrientation orientation) -	{ +{  	if (orientation == LLLayoutStack::HORIZONTAL) -		{ -			F32 collapse_amt =  -			clamp_rescale(mCollapseAmt, 0.f, 1.f, 1.f, (F32)mMinDim / (F32)llmax(1, getRect().getWidth())); -			return mVisibleAmt * collapse_amt; -		} -		else +	{ +		F32 collapse_amt =  +		clamp_rescale(mCollapseAmt, 0.f, 1.f, 1.f, (F32)mMinDim / (F32)llmax(1, getRect().getWidth())); +		return mVisibleAmt * collapse_amt; +	} +	else  	{  			F32 collapse_amt =   			clamp_rescale(mCollapseAmt, 0.f, 1.f, 1.f, llmin(1.f, (F32)mMinDim / (F32)llmax(1, getRect().getHeight())));  			return mVisibleAmt * collapse_amt; -		}  	} +}  //  // LLLayoutStack @@ -109,7 +115,7 @@ LLLayoutStack::LLLayoutStack(const LLLayoutStack::Params& p)  	mMinWidth(0),  	mMinHeight(0),  	mPanelSpacing(p.border_size), -	mOrientation((p.orientation() == "vertical") ? VERTICAL : HORIZONTAL), +	mOrientation(p.orientation),  	mAnimate(p.animate),  	mAnimatedThisFrame(false),  	mClip(p.clip), diff --git a/indra/llui/lllayoutstack.h b/indra/llui/lllayoutstack.h index 2fc6164d7a..4ac8ef0ee9 100644 --- a/indra/llui/lllayoutstack.h +++ b/indra/llui/lllayoutstack.h @@ -37,12 +37,24 @@ class LLLayoutPanel;  class LLLayoutStack : public LLView, public LLInstanceTracker<LLLayoutStack>  {  public: +	typedef enum e_layout_orientation +	{ +		HORIZONTAL, +		VERTICAL +	} ELayoutOrientation; + +	struct OrientationNames +	:	public LLInitParam::TypeValuesHelper<ELayoutOrientation, OrientationNames> +	{ +		static void declareValues(); +	}; +  	struct LayoutStackRegistry : public LLChildRegistry<LayoutStackRegistry>  	{};  	struct Params : public LLInitParam::Block<Params, LLView::Params>  	{ -		Mandatory<std::string>	orientation; +		Mandatory<ELayoutOrientation, OrientationNames>	orientation;  		Optional<S32>			border_size;  		Optional<bool>			animate,  								clip; @@ -54,12 +66,6 @@ public:  	typedef LayoutStackRegistry child_registry_t; -	typedef enum e_layout_orientation -	{ -		HORIZONTAL, -		VERTICAL -	} ELayoutOrientation; -  	virtual ~LLLayoutStack();  	/*virtual*/ void draw(); @@ -171,6 +177,9 @@ public:  	~LLLayoutPanel();  	void initFromParams(const Params& p); +	void setMinDim(S32 value) { mMinDim = value; } +	void setMaxDim(S32 value) { mMaxDim = value; } +  protected:  	LLLayoutPanel(const Params& p)	; diff --git a/indra/llui/lllineeditor.cpp b/indra/llui/lllineeditor.cpp index aed391c780..7e348656a9 100644 --- a/indra/llui/lllineeditor.cpp +++ b/indra/llui/lllineeditor.cpp @@ -88,6 +88,7 @@ LLLineEditor::Params::Params()  	revert_on_esc("revert_on_esc", true),  	commit_on_focus_lost("commit_on_focus_lost", true),  	ignore_tab("ignore_tab", true), +	is_password("is_password", false),  	cursor_color("cursor_color"),  	text_color("text_color"),  	text_readonly_color("text_readonly_color"), @@ -129,7 +130,7 @@ LLLineEditor::LLLineEditor(const LLLineEditor::Params& p)  	mBorderThickness( 0 ),  	mIgnoreArrowKeys( FALSE ),  	mIgnoreTab( p.ignore_tab ), -	mDrawAsterixes( FALSE ), +	mDrawAsterixes( p.is_password ),  	mSelectAllonFocusReceived( p.select_on_focus ),  	mPassDelete(FALSE),  	mReadOnly(FALSE), diff --git a/indra/llui/lllineeditor.h b/indra/llui/lllineeditor.h index ce2dfdeeb8..723423a5b9 100644 --- a/indra/llui/lllineeditor.h +++ b/indra/llui/lllineeditor.h @@ -85,7 +85,8 @@ public:  		Optional<bool>					select_on_focus,  										revert_on_esc,  										commit_on_focus_lost, -										ignore_tab; +										ignore_tab, +										is_password;  		// colors  		Optional<LLUIColor>				cursor_color, diff --git a/indra/llui/llnotifications.cpp b/indra/llui/llnotifications.cpp index c347e15792..cd0f0e36b0 100644 --- a/indra/llui/llnotifications.cpp +++ b/indra/llui/llnotifications.cpp @@ -82,6 +82,7 @@ LLNotificationForm::FormButton::FormButton()  LLNotificationForm::FormInput::FormInput()  :	type("type"), +	text("text"),  	max_length_chars("max_length_chars"),  	width("width", 0),  	value("value") @@ -421,7 +422,7 @@ LLNotificationTemplate::LLNotificationTemplate(const LLNotificationTemplate::Par  		it != end_it;  		++it)  	{ -		mUniqueContext.push_back(it->key); +		mUniqueContext.push_back(it->value);  	}  	lldebugs << "notification \"" << mName << "\": tag count is " << p.tags.size() << llendl; @@ -792,13 +793,19 @@ bool LLNotification::isEquivalentTo(LLNotificationPtr that) const  	{  		const LLSD& these_substitutions = this->getSubstitutions();  		const LLSD& those_substitutions = that->getSubstitutions(); +		const LLSD& this_payload = this->getPayload(); +		const LLSD& that_payload = that->getPayload();  		// highlander bit sez there can only be one of these  		for (std::vector<std::string>::const_iterator it = mTemplatep->mUniqueContext.begin(), end_it = mTemplatep->mUniqueContext.end();  			it != end_it;  			++it)  		{ -			if (these_substitutions.get(*it).asString() != those_substitutions.get(*it).asString()) +			// if templates differ in either substitution strings or payload with the given field name +			// then they are considered inequivalent +			// use of get() avoids converting the LLSD value to a map as the [] operator would +			if (these_substitutions.get(*it).asString() != those_substitutions.get(*it).asString() +				|| this_payload.get(*it).asString() != that_payload.get(*it).asString())  			{  				return false;  			} diff --git a/indra/llui/llnotifications.h b/indra/llui/llnotifications.h index f8f4469958..34d3537781 100644 --- a/indra/llui/llnotifications.h +++ b/indra/llui/llnotifications.h @@ -195,6 +195,7 @@ public:  		Mandatory<std::string>	type;  		Optional<S32>			width;  		Optional<S32>			max_length_chars; +		Optional<std::string>	text;  		Optional<std::string>	value;  		FormInput(); diff --git a/indra/llui/llnotificationtemplate.h b/indra/llui/llnotificationtemplate.h index 5a6ab40a2e..eff572b553 100644 --- a/indra/llui/llnotificationtemplate.h +++ b/indra/llui/llnotificationtemplate.h @@ -74,11 +74,13 @@ struct LLNotificationTemplate  	struct UniquenessContext : public LLInitParam::Block<UniquenessContext>  	{ -		Mandatory<std::string>	key; +		Mandatory<std::string>	value;  		UniquenessContext() -		:	key("key") -		{} +		:	value("value") +		{ +			addSynonym(value, "key"); +		}  	}; @@ -88,7 +90,7 @@ struct LLNotificationTemplate  		// this idiom allows   		// <notification unique="true">  		// as well as -		// <notification> <unique> <context key=""/> </unique>... +		// <notification> <unique> <context></context> </unique>...  		Optional<bool>			dummy_val;  	public:  		Multiple<UniquenessContext>	contexts; @@ -243,8 +245,8 @@ struct LLNotificationTemplate      // (used for things like progress indications, or repeating warnings      // like "the grid is going down in N minutes")      bool mUnique; -    // if we want to be unique only if a certain part of the payload is constant -    // specify the field names for the payload. The notification will only be +    // if we want to be unique only if a certain part of the payload or substitutions args +	// are constant specify the field names for the payload. The notification will only be      // combined if all of the fields named in the context are identical in the      // new and the old notification; otherwise, the notification will be      // duplicated. This is to support suppressing duplicate offers from the same diff --git a/indra/llui/lluictrl.cpp b/indra/llui/lluictrl.cpp index afd60cbb3e..0a06b5e74f 100644 --- a/indra/llui/lluictrl.cpp +++ b/indra/llui/lluictrl.cpp @@ -827,7 +827,7 @@ LLUICtrl* LLUICtrl::findRootMostFocusRoot()  {  	LLUICtrl* focus_root = NULL;  	LLUICtrl* next_view = this; -	while(next_view) +	while(next_view && next_view->hasTabStop())  	{  		if (next_view->isFocusRoot())  		{ diff --git a/indra/llui/llview.h b/indra/llui/llview.h index a5d8e31640..d2bbd663b8 100644 --- a/indra/llui/llview.h +++ b/indra/llui/llview.h @@ -413,14 +413,9 @@ public:  	LLControlVariable *findControl(const std::string& name); -    // Moved setValue(), getValue(), setControlValue(), setControlName(), -    // controlListener() to LLUICtrl because an LLView is NOT assumed to -    // contain a value. If that's what you want, use LLUICtrl instead. -//	virtual bool	handleEvent(LLPointer<LLEvent> event, const LLSD& userdata); -  	const child_list_t*	getChildList() const { return &mChildList; } -	const child_list_const_iter_t	beginChild()  { return mChildList.begin(); } -	const child_list_const_iter_t	endChild()  { return mChildList.end(); } +	child_list_const_iter_t	beginChild() const { return mChildList.begin(); } +	child_list_const_iter_t	endChild() const { return mChildList.end(); }  	// LLMouseHandler functions  	//  Default behavior is to pass events to children diff --git a/indra/llui/llwindowshade.cpp b/indra/llui/llwindowshade.cpp new file mode 100644 index 0000000000..77e94385d4 --- /dev/null +++ b/indra/llui/llwindowshade.cpp @@ -0,0 +1,328 @@ +/** + * @file LLWindowShade.cpp + * @brief Notification dialog that slides down and optionally disabled a piece of UI + * + * $LicenseInfo:firstyear=2006&license=viewerlgpl$ + * Second Life Viewer Source Code + * Copyright (C) 2010, Linden Research, Inc. + *  + * This library is free software; you can redistribute it and/or + * modify it under the terms of the GNU Lesser General Public + * License as published by the Free Software Foundation; + * version 2.1 of the License only. + *  + * This library is distributed in the hope that it will be useful, + * but WITHOUT ANY WARRANTY; without even the implied warranty of + * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU + * Lesser General Public License for more details. + *  + * You should have received a copy of the GNU Lesser General Public + * License along with this library; if not, write to the Free Software + * Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA  02110-1301  USA + *  + * Linden Research, Inc., 945 Battery Street, San Francisco, CA  94111  USA + * $/LicenseInfo$ + */ + +#include "linden_common.h" +#include "llwindowshade.h" + +#include "lllayoutstack.h" +#include "lltextbox.h" +#include "lliconctrl.h" +#include "llbutton.h" +#include "llcheckboxctrl.h" +#include "lllineeditor.h" + +const S32 MIN_NOTIFICATION_AREA_HEIGHT = 30; +const S32 MAX_NOTIFICATION_AREA_HEIGHT = 100; + +LLWindowShade::Params::Params() +:	bg_image("bg_image"), +	modal("modal", false), +	text_color("text_color"), +	can_close("can_close", true) +{ +	mouse_opaque = false; +} + +LLWindowShade::LLWindowShade(const LLWindowShade::Params& params) +:	LLUICtrl(params), +	mNotification(params.notification), +	mModal(params.modal), +	mFormHeight(0), +	mTextColor(params.text_color) +{ +	setFocusRoot(true); +} + +void LLWindowShade::initFromParams(const LLWindowShade::Params& params) +{ +	LLUICtrl::initFromParams(params); + +	LLLayoutStack::Params layout_p; +	layout_p.name = "notification_stack"; +	layout_p.rect = params.rect; +	layout_p.follows.flags = FOLLOWS_ALL; +	layout_p.mouse_opaque = false; +	layout_p.orientation = LLLayoutStack::VERTICAL; +	layout_p.border_size = 0; + +	LLLayoutStack* stackp = LLUICtrlFactory::create<LLLayoutStack>(layout_p); +	addChild(stackp); + +	LLLayoutPanel::Params panel_p; +	panel_p.rect = LLRect(0, 30, 800, 0); +	panel_p.name = "notification_area"; +	panel_p.visible = false; +	panel_p.user_resize = false; +	panel_p.background_visible = true; +	panel_p.bg_alpha_image = params.bg_image; +	panel_p.auto_resize = false; +	LLLayoutPanel* notification_panel = LLUICtrlFactory::create<LLLayoutPanel>(panel_p); +	stackp->addChild(notification_panel); + +	panel_p = LLUICtrlFactory::getDefaultParams<LLLayoutPanel>(); +	panel_p.auto_resize = true; +	panel_p.user_resize = false; +	panel_p.rect = params.rect; +	panel_p.name = "background_area"; +	panel_p.mouse_opaque = false; +	panel_p.background_visible = false; +	panel_p.bg_alpha_color = LLColor4(0.f, 0.f, 0.f, 0.2f); +	LLLayoutPanel* dummy_panel = LLUICtrlFactory::create<LLLayoutPanel>(panel_p); +	stackp->addChild(dummy_panel); + +	layout_p = LLUICtrlFactory::getDefaultParams<LLLayoutStack>(); +	layout_p.rect = LLRect(0, 30, 800, 0); +	layout_p.follows.flags = FOLLOWS_ALL; +	layout_p.orientation = LLLayoutStack::HORIZONTAL; +	stackp = LLUICtrlFactory::create<LLLayoutStack>(layout_p); +	notification_panel->addChild(stackp); + +	panel_p = LLUICtrlFactory::getDefaultParams<LLLayoutPanel>(); +	panel_p.rect.height = 30; +	LLLayoutPanel* panel = LLUICtrlFactory::create<LLLayoutPanel>(panel_p); +	stackp->addChild(panel); + +	LLIconCtrl::Params icon_p; +	icon_p.name = "notification_icon"; +	icon_p.rect = LLRect(5, 23, 21, 8); +	panel->addChild(LLUICtrlFactory::create<LLIconCtrl>(icon_p)); + +	LLTextBox::Params text_p; +	text_p.rect = LLRect(31, 20, panel->getRect().getWidth() - 5, 0); +	text_p.follows.flags = FOLLOWS_ALL; +	text_p.text_color = mTextColor; +	text_p.font = LLFontGL::getFontSansSerifSmall(); +	text_p.font.style = "BOLD"; +	text_p.name = "notification_text"; +	text_p.use_ellipses = true; +	text_p.wrap = true; +	panel->addChild(LLUICtrlFactory::create<LLTextBox>(text_p)); + +	panel_p = LLUICtrlFactory::getDefaultParams<LLLayoutPanel>(); +	panel_p.auto_resize = false; +	panel_p.user_resize = false; +	panel_p.name="form_elements"; +	panel_p.rect = LLRect(0, 30, 130, 0); +	LLLayoutPanel* form_elements_panel = LLUICtrlFactory::create<LLLayoutPanel>(panel_p); +	stackp->addChild(form_elements_panel); + +	if (params.can_close) +	{ +		panel_p = LLUICtrlFactory::getDefaultParams<LLLayoutPanel>(); +		panel_p.auto_resize = false; +		panel_p.user_resize = false; +		panel_p.rect = LLRect(0, 30, 25, 0); +		LLLayoutPanel* close_panel = LLUICtrlFactory::create<LLLayoutPanel>(panel_p); +		stackp->addChild(close_panel); + +		LLButton::Params button_p; +		button_p.name = "close_notification"; +		button_p.rect = LLRect(5, 23, 21, 7); +		button_p.image_color.control="DkGray_66"; +		button_p.image_unselected.name="Icon_Close_Foreground"; +		button_p.image_selected.name="Icon_Close_Press"; +		button_p.click_callback.function = boost::bind(&LLWindowShade::onCloseNotification, this); + +		close_panel->addChild(LLUICtrlFactory::create<LLButton>(button_p)); +	} + +	LLSD payload = mNotification->getPayload(); + +	LLNotificationFormPtr formp = mNotification->getForm(); +	LLLayoutPanel& notification_area = getChildRef<LLLayoutPanel>("notification_area"); +	notification_area.getChild<LLUICtrl>("notification_icon")->setValue(mNotification->getIcon()); +	notification_area.getChild<LLUICtrl>("notification_text")->setValue(mNotification->getMessage()); +	notification_area.getChild<LLUICtrl>("notification_text")->setToolTip(mNotification->getMessage()); + +	LLNotificationForm::EIgnoreType ignore_type = formp->getIgnoreType();  +	LLLayoutPanel& form_elements = notification_area.getChildRef<LLLayoutPanel>("form_elements"); +	form_elements.deleteAllChildren(); + +	const S32 FORM_PADDING_HORIZONTAL = 10; +	const S32 FORM_PADDING_VERTICAL = 3; +	const S32 WIDGET_HEIGHT = 24; +	const S32 LINE_EDITOR_WIDTH = 120; +	S32 cur_x = FORM_PADDING_HORIZONTAL; +	S32 cur_y = FORM_PADDING_VERTICAL + WIDGET_HEIGHT; +	S32 form_width = cur_x; + +	if (ignore_type != LLNotificationForm::IGNORE_NO) +	{ +		LLCheckBoxCtrl::Params checkbox_p; +		checkbox_p.name = "ignore_check"; +		checkbox_p.rect = LLRect(cur_x, cur_y, cur_x, cur_y - WIDGET_HEIGHT); +		checkbox_p.label = formp->getIgnoreMessage(); +		checkbox_p.label_text.text_color = LLColor4::black; +		checkbox_p.commit_callback.function = boost::bind(&LLWindowShade::onClickIgnore, this, _1); +		checkbox_p.initial_value = formp->getIgnored(); + +		LLCheckBoxCtrl* check = LLUICtrlFactory::create<LLCheckBoxCtrl>(checkbox_p); +		check->setRect(check->getBoundingRect()); +		form_elements.addChild(check); +		cur_x = check->getRect().mRight + FORM_PADDING_HORIZONTAL; +		form_width = llmax(form_width, cur_x); +	} + +	for (S32 i = 0; i < formp->getNumElements(); i++) +	{ +		LLSD form_element = formp->getElement(i); +		std::string type = form_element["type"].asString(); +		if (type == "button") +		{ +			LLButton::Params button_p; +			button_p.name = form_element["name"]; +			button_p.label = form_element["text"]; +			button_p.rect = LLRect(cur_x, cur_y, cur_x, cur_y - WIDGET_HEIGHT); +			button_p.click_callback.function = boost::bind(&LLWindowShade::onClickNotificationButton, this, form_element["name"].asString()); +			button_p.auto_resize = true; + +			LLButton* button = LLUICtrlFactory::create<LLButton>(button_p); +			button->autoResize(); +			form_elements.addChild(button); + +			if (form_element["default"].asBoolean()) +			{ +				form_elements.setDefaultBtn(button); +			} + +			cur_x = button->getRect().mRight + FORM_PADDING_HORIZONTAL; +			form_width = llmax(form_width, cur_x); +		} +		else if (type == "text" || type == "password") +		{ +			// if not at beginning of line... +			if (cur_x != FORM_PADDING_HORIZONTAL) +			{ +				// start new line +				cur_x = FORM_PADDING_HORIZONTAL; +				cur_y -= WIDGET_HEIGHT + FORM_PADDING_VERTICAL; +			} +			LLTextBox::Params label_p; +			label_p.name = form_element["name"].asString() + "_label"; +			label_p.rect = LLRect(cur_x, cur_y, cur_x + LINE_EDITOR_WIDTH, cur_y - WIDGET_HEIGHT); +			label_p.initial_value = form_element["text"]; +			label_p.text_color = mTextColor; +			label_p.font_valign = LLFontGL::VCENTER; +			label_p.v_pad = 5; +			LLTextBox* textbox = LLUICtrlFactory::create<LLTextBox>(label_p); +			textbox->reshapeToFitText(); +			textbox->reshape(textbox->getRect().getWidth(), form_elements.getRect().getHeight() - 2 * FORM_PADDING_VERTICAL);  +			form_elements.addChild(textbox); +			cur_x = textbox->getRect().mRight + FORM_PADDING_HORIZONTAL; + +			LLLineEditor::Params line_p; +			line_p.name = form_element["name"]; +			line_p.keystroke_callback = boost::bind(&LLWindowShade::onEnterNotificationText, this, _1, form_element["name"].asString()); +			line_p.is_password = type == "password"; +			line_p.rect = LLRect(cur_x, cur_y, cur_x + LINE_EDITOR_WIDTH, cur_y - WIDGET_HEIGHT); + +			LLLineEditor* line_editor = LLUICtrlFactory::create<LLLineEditor>(line_p); +			form_elements.addChild(line_editor); +			form_width = llmax(form_width, cur_x + LINE_EDITOR_WIDTH + FORM_PADDING_HORIZONTAL); + +			// reset to start of next line +			cur_x = FORM_PADDING_HORIZONTAL; +			cur_y -= WIDGET_HEIGHT + FORM_PADDING_VERTICAL; +		} +	} + +	mFormHeight = form_elements.getRect().getHeight() - (cur_y - FORM_PADDING_VERTICAL) + WIDGET_HEIGHT; +	form_elements.reshape(form_width, mFormHeight); +	form_elements.setMinDim(form_width); + +	// move all form elements back onto form surface +	S32 delta_y = WIDGET_HEIGHT + FORM_PADDING_VERTICAL - cur_y; +	for (child_list_const_iter_t it = form_elements.getChildList()->begin(), end_it = form_elements.getChildList()->end(); +		it != end_it; +		++it) +	{ +		(*it)->translate(0, delta_y); +	} +} + +void LLWindowShade::show() +{ +	getChildRef<LLLayoutPanel>("notification_area").setVisible(true); +	getChildRef<LLLayoutPanel>("background_area").setBackgroundVisible(mModal); + +	setMouseOpaque(mModal); +} + +void LLWindowShade::draw() +{ +	LLRect message_rect = getChild<LLTextBox>("notification_text")->getTextBoundingRect(); + +	LLLayoutPanel* notification_area = getChild<LLLayoutPanel>("notification_area"); + +	notification_area->reshape(notification_area->getRect().getWidth(),  +		llclamp(message_rect.getHeight() + 10,  +				llmin(mFormHeight, MAX_NOTIFICATION_AREA_HEIGHT), +				MAX_NOTIFICATION_AREA_HEIGHT)); + +	LLUICtrl::draw(); +	if (mNotification && !mNotification->isActive()) +	{ +		hide(); +	} +} + +void LLWindowShade::hide() +{ +	getChildRef<LLLayoutPanel>("notification_area").setVisible(false); +	getChildRef<LLLayoutPanel>("background_area").setBackgroundVisible(false); + +	setMouseOpaque(false); +} + +void LLWindowShade::onCloseNotification() +{ +	LLNotifications::instance().cancel(mNotification); +} + +void LLWindowShade::onClickIgnore(LLUICtrl* ctrl) +{ +	bool check = ctrl->getValue().asBoolean(); +	if (mNotification && mNotification->getForm()->getIgnoreType() == LLNotificationForm::IGNORE_SHOW_AGAIN) +	{ +		// question was "show again" so invert value to get "ignore" +		check = !check; +	} +	mNotification->setIgnored(check); +} + +void LLWindowShade::onClickNotificationButton(const std::string& name) +{ +	if (!mNotification) return; + +	mNotificationResponse[name] = true; + +	mNotification->respond(mNotificationResponse); +} + +void LLWindowShade::onEnterNotificationText(LLUICtrl* ctrl, const std::string& name) +{ +	mNotificationResponse[name] = ctrl->getValue().asString(); +} diff --git a/indra/llui/llwindowshade.h b/indra/llui/llwindowshade.h new file mode 100644 index 0000000000..0047195929 --- /dev/null +++ b/indra/llui/llwindowshade.h @@ -0,0 +1,69 @@ +/** + * @file llwindowshade.h + * @brief Notification dialog that slides down and optionally disabled a piece of UI + * + * $LicenseInfo:firstyear=2006&license=viewerlgpl$ + * Second Life Viewer Source Code + * Copyright (C) 2010, Linden Research, Inc. + *  + * This library is free software; you can redistribute it and/or + * modify it under the terms of the GNU Lesser General Public + * License as published by the Free Software Foundation; + * version 2.1 of the License only. + *  + * This library is distributed in the hope that it will be useful, + * but WITHOUT ANY WARRANTY; without even the implied warranty of + * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU + * Lesser General Public License for more details. + *  + * You should have received a copy of the GNU Lesser General Public + * License along with this library; if not, write to the Free Software + * Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA  02110-1301  USA + *  + * Linden Research, Inc., 945 Battery Street, San Francisco, CA  94111  USA + * $/LicenseInfo$ + */ + +#ifndef LL_LLWINDOWSHADE_H +#define LL_LLWINDOWSHADE_H + +#include "lluictrl.h" +#include "llnotifications.h" + +class LLWindowShade : public LLUICtrl +{ +public: +	struct Params : public LLInitParam::Block<Params, LLUICtrl::Params> +	{ +		Mandatory<LLNotificationPtr>	notification; +		Optional<LLUIImage*>			bg_image; +		Optional<LLUIColor>				text_color; +		Optional<bool>					modal, +										can_close; + +		Params(); +	}; + +	void show(); +	/*virtual*/ void draw(); +	void hide(); + +private: +	friend class LLUICtrlFactory; + +	LLWindowShade(const Params& p); +	void initFromParams(const Params& params); + +	void onCloseNotification(); +	void onClickNotificationButton(const std::string& name); +	void onEnterNotificationText(LLUICtrl* ctrl, const std::string& name); +	void onClickIgnore(LLUICtrl* ctrl); + +	LLNotificationPtr	mNotification; +	LLSD				mNotificationResponse; +	bool				mModal; +	S32					mFormHeight; +	LLUIColor			mTextColor; +}; + +#endif // LL_LLWINDOWSHADE_H diff --git a/indra/media_plugins/example/media_plugin_example.cpp b/indra/media_plugins/example/media_plugin_example.cpp index f8a871930e..da7de01799 100644 --- a/indra/media_plugins/example/media_plugin_example.cpp +++ b/indra/media_plugins/example/media_plugin_example.cpp @@ -6,21 +6,21 @@   * $LicenseInfo:firstyear=2008&license=viewerlgpl$   * Second Life Viewer Source Code   * Copyright (C) 2010, Linden Research, Inc. - *  + *   * This library is free software; you can redistribute it and/or   * modify it under the terms of the GNU Lesser General Public   * License as published by the Free Software Foundation;   * version 2.1 of the License only. - *  + *   * This library is distributed in the hope that it will be useful,   * but WITHOUT ANY WARRANTY; without even the implied warranty of   * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU   * Lesser General Public License for more details. - *  + *   * You should have received a copy of the GNU Lesser General Public   * License along with this library; if not, write to the Free Software   * Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA  02110-1301  USA - *  + *   * Linden Research, Inc., 945 Battery Street, San Francisco, CA  94111  USA   * $/LicenseInfo$   * @endcond @@ -39,48 +39,48 @@  ////////////////////////////////////////////////////////////////////////////////  //  class MediaPluginExample : -		public MediaPluginBase +        public MediaPluginBase  { -	public: -		MediaPluginExample( LLPluginInstance::sendMessageFunction host_send_func, void *host_user_data ); -		~MediaPluginExample(); - -		/*virtual*/ void receiveMessage( const char* message_string ); - -	private: -		bool init(); -		void update( F64 milliseconds ); -		void write_pixel( int x, int y, unsigned char r, unsigned char g, unsigned char b ); -		bool mFirstTime; - -		time_t mLastUpdateTime; -		enum Constants { ENumObjects = 10 }; -		unsigned char* mBackgroundPixels; -		int mColorR[ ENumObjects ]; -		int mColorG[ ENumObjects ]; -		int mColorB[ ENumObjects ]; -		int mXpos[ ENumObjects ]; -		int mYpos[ ENumObjects ]; -		int mXInc[ ENumObjects ]; -		int mYInc[ ENumObjects ]; -		int mBlockSize[ ENumObjects ]; -		bool mMouseButtonDown; -		bool mStopAction; +    public: +        MediaPluginExample( LLPluginInstance::sendMessageFunction host_send_func, void *host_user_data ); +        ~MediaPluginExample(); + +        /*virtual*/ void receiveMessage( const char* message_string ); + +    private: +        bool init(); +        void update( F64 milliseconds ); +        void write_pixel( int x, int y, unsigned char r, unsigned char g, unsigned char b ); +        bool mFirstTime; + +        time_t mLastUpdateTime; +        enum Constants { ENumObjects = 10 }; +        unsigned char* mBackgroundPixels; +        int mColorR[ ENumObjects ]; +        int mColorG[ ENumObjects ]; +        int mColorB[ ENumObjects ]; +        int mXpos[ ENumObjects ]; +        int mYpos[ ENumObjects ]; +        int mXInc[ ENumObjects ]; +        int mYInc[ ENumObjects ]; +        int mBlockSize[ ENumObjects ]; +        bool mMouseButtonDown; +        bool mStopAction;  };  ////////////////////////////////////////////////////////////////////////////////  //  MediaPluginExample::MediaPluginExample( LLPluginInstance::sendMessageFunction host_send_func, void *host_user_data ) : -	MediaPluginBase( host_send_func, host_user_data ) +    MediaPluginBase( host_send_func, host_user_data )  { -	mFirstTime = true; -	mWidth = 0; -	mHeight = 0; -	mDepth = 4; -	mPixels = 0; -	mMouseButtonDown = false; -	mStopAction = false; -	mLastUpdateTime = 0; +    mFirstTime = true; +    mWidth = 0; +    mHeight = 0; +    mDepth = 4; +    mPixels = 0; +    mMouseButtonDown = false; +    mStopAction = false; +    mLastUpdateTime = 0;  }  //////////////////////////////////////////////////////////////////////////////// @@ -93,395 +93,320 @@ MediaPluginExample::~MediaPluginExample()  //  void MediaPluginExample::receiveMessage( const char* message_string )  { -	LLPluginMessage message_in; - -	if ( message_in.parse( message_string ) >= 0 ) -	{ -		std::string message_class = message_in.getClass(); -		std::string message_name = message_in.getName(); - -		if ( message_class == LLPLUGIN_MESSAGE_CLASS_BASE ) -		{ -			if ( message_name == "init" ) -			{ -				LLPluginMessage message( "base", "init_response" ); -				LLSD versions = LLSD::emptyMap(); -				versions[ LLPLUGIN_MESSAGE_CLASS_BASE ] = LLPLUGIN_MESSAGE_CLASS_BASE_VERSION; -				versions[ LLPLUGIN_MESSAGE_CLASS_MEDIA ] = LLPLUGIN_MESSAGE_CLASS_MEDIA_VERSION; -				versions[ LLPLUGIN_MESSAGE_CLASS_MEDIA_BROWSER ] = LLPLUGIN_MESSAGE_CLASS_MEDIA_BROWSER_VERSION; -				message.setValueLLSD( "versions", versions ); - -				std::string plugin_version = "Example media plugin, Example Version 1.0.0.0"; -				message.setValue( "plugin_version", plugin_version ); -				sendMessage( message ); -			} -			else -			if ( message_name == "idle" ) -			{ -				// no response is necessary here. -				F64 time = message_in.getValueReal( "time" ); - -				// Convert time to milliseconds for update() -				update( time ); -			} -			else -			if ( message_name == "cleanup" ) -			{ -				// clean up here -			} -			else -			if ( message_name == "shm_added" ) -			{ -				SharedSegmentInfo info; -				info.mAddress = message_in.getValuePointer( "address" ); -				info.mSize = ( size_t )message_in.getValueS32( "size" ); -				std::string name = message_in.getValue( "name" ); - -				mSharedSegments.insert( SharedSegmentMap::value_type( name, info ) ); - -			} -			else -			if ( message_name == "shm_remove" ) -			{ -				std::string name = message_in.getValue( "name" ); - -				SharedSegmentMap::iterator iter = mSharedSegments.find( name ); -				if( iter != mSharedSegments.end() ) -				{ -					if ( mPixels == iter->second.mAddress ) -					{ -						// This is the currently active pixel buffer. -						// Make sure we stop drawing to it. -						mPixels = NULL; -						mTextureSegmentName.clear(); -					}; -					mSharedSegments.erase( iter ); -				} -				else -				{ -					//std::cerr << "MediaPluginExample::receiveMessage: unknown shared memory region!" << std::endl; -				}; - -				// Send the response so it can be cleaned up. -				LLPluginMessage message( "base", "shm_remove_response" ); -				message.setValue( "name", name ); -				sendMessage( message ); -			} -			else -			{ -				//std::cerr << "MediaPluginExample::receiveMessage: unknown base message: " << message_name << std::endl; -			}; -		} -		else -		if ( message_class == LLPLUGIN_MESSAGE_CLASS_MEDIA ) -		{ -			if ( message_name == "init" ) -			{ -				// Plugin gets to decide the texture parameters to use. -				LLPluginMessage message( LLPLUGIN_MESSAGE_CLASS_MEDIA, "texture_params" ); -				message.setValueS32( "default_width", mWidth ); -				message.setValueS32( "default_height", mHeight ); -				message.setValueS32( "depth", mDepth ); -				message.setValueU32( "internalformat", GL_RGBA ); -				message.setValueU32( "format", GL_RGBA ); -				message.setValueU32( "type", GL_UNSIGNED_BYTE ); -				message.setValueBoolean( "coords_opengl", false ); -				sendMessage( message ); -			} -			else if ( message_name == "size_change" ) -			{ -				std::string name = message_in.getValue( "name" ); -				S32 width = message_in.getValueS32( "width" ); -				S32 height = message_in.getValueS32( "height" ); -				S32 texture_width = message_in.getValueS32( "texture_width" ); -				S32 texture_height = message_in.getValueS32( "texture_height" ); - -				if ( ! name.empty() ) -				{ -					// Find the shared memory region with this name -					SharedSegmentMap::iterator iter = mSharedSegments.find( name ); -					if ( iter != mSharedSegments.end() ) -					{ -						mPixels = ( unsigned char* )iter->second.mAddress; -						mWidth = width; -						mHeight = height; - -						mTextureWidth = texture_width; -						mTextureHeight = texture_height; - -						init(); -					}; -				}; - -				LLPluginMessage message( LLPLUGIN_MESSAGE_CLASS_MEDIA, "size_change_response" ); -				message.setValue( "name", name ); -				message.setValueS32( "width", width ); -				message.setValueS32( "height", height ); -				message.setValueS32( "texture_width", texture_width ); -				message.setValueS32( "texture_height", texture_height ); -				sendMessage( message ); -			} -			else -			if ( message_name == "load_uri" ) -			{ -				std::string uri = message_in.getValue( "uri" ); -				if ( ! uri.empty() ) -				{ -				}; -			} -			else -			if ( message_name == "mouse_event" ) -			{ -				std::string event = message_in.getValue( "event" ); -				S32 button = message_in.getValueS32( "button" ); - -				// left mouse button -				if ( button == 0 ) -				{ -					int mouse_x = message_in.getValueS32( "x" ); -					int mouse_y = message_in.getValueS32( "y" ); -					std::string modifiers = message_in.getValue( "modifiers" ); - -					if ( event == "move" ) -					{ -						if ( mMouseButtonDown ) -							write_pixel( mouse_x, mouse_y, rand() % 0x80 + 0x80, rand() % 0x80 + 0x80, rand() % 0x80 + 0x80 ); -					} -					else -					if ( event == "down" ) -					{ -						mMouseButtonDown = true; -					} -					else -					if ( event == "up" ) -					{ -						mMouseButtonDown = false; -					} -					else -					if ( event == "double_click" ) -					{ -					}; -				}; -			} -			else -			if ( message_name == "key_event" ) -			{ -				std::string event = message_in.getValue( "event" ); -				S32 key = message_in.getValueS32( "key" ); -				std::string modifiers = message_in.getValue( "modifiers" ); - -				if ( event == "down" ) -				{ -					if ( key == ' ') -					{ -						mLastUpdateTime = 0; -						update( 0.0f ); -					}; -				}; -			} -			else -			{ -				//std::cerr << "MediaPluginExample::receiveMessage: unknown media message: " << message_string << std::endl; -			}; -		} -		else -		if ( message_class == LLPLUGIN_MESSAGE_CLASS_MEDIA_BROWSER ) -		{ -			if ( message_name == "browse_reload" ) -			{ -				mLastUpdateTime = 0; -				mFirstTime = true; -				mStopAction = false; -				update( 0.0f ); -			} -			else -			if ( message_name == "browse_stop" ) -			{ -				for( int n = 0; n < ENumObjects; ++n ) -					mXInc[ n ] = mYInc[ n ] = 0; - -				mStopAction = true; -				update( 0.0f ); -			} -			else -			{ -				//std::cerr << "MediaPluginExample::receiveMessage: unknown media_browser message: " << message_string << std::endl; -			}; -		} -		else -		{ -			//std::cerr << "MediaPluginExample::receiveMessage: unknown message class: " << message_class << std::endl; -		}; -	}; +//  std::cerr << "MediaPluginWebKit::receiveMessage: received message: \"" << message_string << "\"" << std::endl; +    LLPluginMessage message_in; + +    if(message_in.parse(message_string) >= 0) +    { +        std::string message_class = message_in.getClass(); +        std::string message_name = message_in.getName(); +        if(message_class == LLPLUGIN_MESSAGE_CLASS_BASE) +        { +            if(message_name == "init") +            { +                LLPluginMessage message("base", "init_response"); +                LLSD versions = LLSD::emptyMap(); +                versions[LLPLUGIN_MESSAGE_CLASS_BASE] = LLPLUGIN_MESSAGE_CLASS_BASE_VERSION; +                versions[LLPLUGIN_MESSAGE_CLASS_MEDIA] = LLPLUGIN_MESSAGE_CLASS_MEDIA_VERSION; +                versions[LLPLUGIN_MESSAGE_CLASS_MEDIA_BROWSER] = LLPLUGIN_MESSAGE_CLASS_MEDIA_BROWSER_VERSION; +                message.setValueLLSD("versions", versions); + +                std::string plugin_version = "Example plugin 1.0..0"; +                message.setValue("plugin_version", plugin_version); +                sendMessage(message); +            } +            else if(message_name == "idle") +            { +                // no response is necessary here. +                F64 time = message_in.getValueReal("time"); + +                // Convert time to milliseconds for update() +                update((int)(time * 1000.0f)); +            } +            else if(message_name == "cleanup") +            { +            } +            else if(message_name == "shm_added") +            { +                SharedSegmentInfo info; +                info.mAddress = message_in.getValuePointer("address"); +                info.mSize = (size_t)message_in.getValueS32("size"); +                std::string name = message_in.getValue("name"); + +                mSharedSegments.insert(SharedSegmentMap::value_type(name, info)); + +            } +            else if(message_name == "shm_remove") +            { +                std::string name = message_in.getValue("name"); + +                SharedSegmentMap::iterator iter = mSharedSegments.find(name); +                if(iter != mSharedSegments.end()) +                { +                    if(mPixels == iter->second.mAddress) +                    { +                        // This is the currently active pixel buffer.  Make sure we stop drawing to it. +                        mPixels = NULL; +                        mTextureSegmentName.clear(); +                    } +                    mSharedSegments.erase(iter); +                } +                else +                { +//                  std::cerr << "MediaPluginWebKit::receiveMessage: unknown shared memory region!" << std::endl; +                } + +                // Send the response so it can be cleaned up. +                LLPluginMessage message("base", "shm_remove_response"); +                message.setValue("name", name); +                sendMessage(message); +            } +            else +            { +//              std::cerr << "MediaPluginWebKit::receiveMessage: unknown base message: " << message_name << std::endl; +            } +        } +        else if(message_class == LLPLUGIN_MESSAGE_CLASS_MEDIA) +        { +            if(message_name == "init") +            { +                // Plugin gets to decide the texture parameters to use. +                mDepth = 4; +                LLPluginMessage message(LLPLUGIN_MESSAGE_CLASS_MEDIA, "texture_params"); +                message.setValueS32("default_width", 1024); +                message.setValueS32("default_height", 1024); +                message.setValueS32("depth", mDepth); +                message.setValueU32("internalformat", GL_RGBA); +                message.setValueU32("format", GL_RGBA); +                message.setValueU32("type", GL_UNSIGNED_BYTE); +                message.setValueBoolean("coords_opengl", true); +                sendMessage(message); +            } +            else if(message_name == "size_change") +            { +                std::string name = message_in.getValue("name"); +                S32 width = message_in.getValueS32("width"); +                S32 height = message_in.getValueS32("height"); +                S32 texture_width = message_in.getValueS32("texture_width"); +                S32 texture_height = message_in.getValueS32("texture_height"); + +                if(!name.empty()) +                { +                    // Find the shared memory region with this name +                    SharedSegmentMap::iterator iter = mSharedSegments.find(name); +                    if(iter != mSharedSegments.end()) +                    { +                        mPixels = (unsigned char*)iter->second.mAddress; +                        mWidth = width; +                        mHeight = height; + +                        mTextureWidth = texture_width; +                        mTextureHeight = texture_height; +                    }; +                }; + +                LLPluginMessage message(LLPLUGIN_MESSAGE_CLASS_MEDIA, "size_change_response"); +                message.setValue("name", name); +                message.setValueS32("width", width); +                message.setValueS32("height", height); +                message.setValueS32("texture_width", texture_width); +                message.setValueS32("texture_height", texture_height); +                sendMessage(message); + +            } +            else if(message_name == "load_uri") +            { +            } +            else if(message_name == "mouse_event") +            { +                std::string event = message_in.getValue("event"); +                if(event == "down") +                { + +                } +                else if(event == "up") +                { +                } +                else if(event == "double_click") +                { +                } +            } +        } +        else +        { +//          std::cerr << "MediaPluginWebKit::receiveMessage: unknown message class: " << message_class << std::endl; +        }; +    }  }  ////////////////////////////////////////////////////////////////////////////////  //  void MediaPluginExample::write_pixel( int x, int y, unsigned char r, unsigned char g, unsigned char b )  { -	// make sure we don't write outside the buffer -	if ( ( x < 0 ) || ( x >= mWidth ) || ( y < 0 ) || ( y >= mHeight ) ) -		return; -		 -	if ( mBackgroundPixels != NULL ) -	{ -		unsigned char *pixel = mBackgroundPixels; -		pixel += y * mWidth * mDepth; -		pixel += ( x * mDepth ); -		pixel[ 0 ] = b; -		pixel[ 1 ] = g; -		pixel[ 2 ] = r; - -		setDirty( x, y, x + 1, y + 1 ); -	}; +    // make sure we don't write outside the buffer +    if ( ( x < 0 ) || ( x >= mWidth ) || ( y < 0 ) || ( y >= mHeight ) ) +        return; + +    if ( mBackgroundPixels != NULL ) +    { +        unsigned char *pixel = mBackgroundPixels; +        pixel += y * mWidth * mDepth; +        pixel += ( x * mDepth ); +        pixel[ 0 ] = b; +        pixel[ 1 ] = g; +        pixel[ 2 ] = r; + +        setDirty( x, y, x + 1, y + 1 ); +    };  }  ////////////////////////////////////////////////////////////////////////////////  //  void MediaPluginExample::update( F64 milliseconds )  { -	if ( mWidth < 1 || mWidth > 2048 || mHeight < 1 || mHeight > 2048 ) -		return; - -	if ( mPixels == 0 ) -			return; - -	if ( mFirstTime ) -	{ -		for( int n = 0; n < ENumObjects; ++n ) -		{ -			mXpos[ n ] = ( mWidth / 2 ) + rand() % ( mWidth / 16 ) - ( mWidth / 32 ); -			mYpos[ n ] = ( mHeight / 2 ) + rand() % ( mHeight / 16 ) - ( mHeight / 32 ); - -			mColorR[ n ] = rand() % 0x60 + 0x60; -			mColorG[ n ] = rand() % 0x60 + 0x60; -			mColorB[ n ] = rand() % 0x60 + 0x60; - -			mXInc[ n ] = 0; -			while ( mXInc[ n ] == 0 ) -				mXInc[ n ] = rand() % 7 - 3; - -			mYInc[ n ] = 0; -			while ( mYInc[ n ] == 0 ) -				mYInc[ n ] = rand() % 9 - 4; - -			mBlockSize[ n ] = rand() % 0x30 + 0x10; -		}; - -		delete [] mBackgroundPixels; -				 -		mBackgroundPixels = new unsigned char[ mWidth * mHeight * mDepth ]; - -		mFirstTime = false; -	}; - -	if ( mStopAction ) -		return; - -	if ( time( NULL ) > mLastUpdateTime + 3 ) -	{ -		const int num_squares = rand() % 20 + 4; -		int sqr1_r = rand() % 0x80 + 0x20; -		int sqr1_g = rand() % 0x80 + 0x20; -		int sqr1_b = rand() % 0x80 + 0x20; -		int sqr2_r = rand() % 0x80 + 0x20; -		int sqr2_g = rand() % 0x80 + 0x20; -		int sqr2_b = rand() % 0x80 + 0x20; - -		for ( int y1 = 0; y1 < num_squares; ++y1 ) -		{ -			for ( int x1 = 0; x1 < num_squares; ++x1 ) -			{ -				int px_start = mWidth * x1 / num_squares; -				int px_end = ( mWidth * ( x1 + 1 ) ) / num_squares; -				int py_start = mHeight * y1 / num_squares; -				int py_end = ( mHeight * ( y1 + 1 ) ) / num_squares; - -				for( int y2 = py_start; y2 < py_end; ++y2 ) -				{ -					for( int x2 = px_start; x2 < px_end; ++x2 ) -					{ -						int rowspan = mWidth * mDepth; - -						if ( ( y1 % 2 ) ^ ( x1 % 2 ) ) -						{ -							mBackgroundPixels[ y2 * rowspan + x2 * mDepth + 0 ] = sqr1_r; -							mBackgroundPixels[ y2 * rowspan + x2 * mDepth + 1 ] = sqr1_g; -							mBackgroundPixels[ y2 * rowspan + x2 * mDepth + 2 ] = sqr1_b; -						} -						else -						{ -							mBackgroundPixels[ y2 * rowspan + x2 * mDepth + 0 ] = sqr2_r; -							mBackgroundPixels[ y2 * rowspan + x2 * mDepth + 1 ] = sqr2_g; -							mBackgroundPixels[ y2 * rowspan + x2 * mDepth + 2 ] = sqr2_b; -						}; -					}; -				}; -			}; -		}; - -		time( &mLastUpdateTime ); -	}; - -	memcpy( mPixels, mBackgroundPixels, mWidth * mHeight * mDepth ); - -	for( int n = 0; n < ENumObjects; ++n ) -	{ -		if ( rand() % 50 == 0 ) -		{ -				mXInc[ n ] = 0; -				while ( mXInc[ n ] == 0 ) -					mXInc[ n ] = rand() % 7 - 3; - -				mYInc[ n ] = 0; -				while ( mYInc[ n ] == 0 ) -					mYInc[ n ] = rand() % 9 - 4; -		}; - -		if ( mXpos[ n ] + mXInc[ n ] < 0 || mXpos[ n ] + mXInc[ n ] >= mWidth - mBlockSize[ n ] ) -			mXInc[ n ] =- mXInc[ n ]; - -		if ( mYpos[ n ] + mYInc[ n ] < 0 || mYpos[ n ] + mYInc[ n ] >= mHeight - mBlockSize[ n ] ) -			mYInc[ n ] =- mYInc[ n ]; - -		mXpos[ n ] += mXInc[ n ]; -		mYpos[ n ] += mYInc[ n ]; - -		for( int y = 0; y < mBlockSize[ n ]; ++y ) -		{ -			for( int x = 0; x < mBlockSize[ n ]; ++x ) -			{ -				mPixels[ ( mXpos[ n ] + x ) * mDepth + ( mYpos[ n ] + y ) * mDepth * mWidth + 0 ] = mColorR[ n ]; -				mPixels[ ( mXpos[ n ] + x ) * mDepth + ( mYpos[ n ] + y ) * mDepth * mWidth + 1 ] = mColorG[ n ]; -				mPixels[ ( mXpos[ n ] + x ) * mDepth + ( mYpos[ n ] + y ) * mDepth * mWidth + 2 ] = mColorB[ n ]; -			}; -		}; -	}; - -	setDirty( 0, 0, mWidth, mHeight ); +    if ( mWidth < 1 || mWidth > 2048 || mHeight < 1 || mHeight > 2048 ) +        return; + +    if ( mPixels == 0 ) +            return; + +    if ( mFirstTime ) +    { +        for( int n = 0; n < ENumObjects; ++n ) +        { +            mXpos[ n ] = ( mWidth / 2 ) + rand() % ( mWidth / 16 ) - ( mWidth / 32 ); +            mYpos[ n ] = ( mHeight / 2 ) + rand() % ( mHeight / 16 ) - ( mHeight / 32 ); + +            mColorR[ n ] = rand() % 0x60 + 0x60; +            mColorG[ n ] = rand() % 0x60 + 0x60; +            mColorB[ n ] = rand() % 0x60 + 0x60; + +            mXInc[ n ] = 0; +            while ( mXInc[ n ] == 0 ) +                mXInc[ n ] = rand() % 7 - 3; + +            mYInc[ n ] = 0; +            while ( mYInc[ n ] == 0 ) +                mYInc[ n ] = rand() % 9 - 4; + +            mBlockSize[ n ] = rand() % 0x30 + 0x10; +        }; + +        delete [] mBackgroundPixels; + +        mBackgroundPixels = new unsigned char[ mWidth * mHeight * mDepth ]; + +        mFirstTime = false; +    }; + +    if ( mStopAction ) +        return; + +    if ( time( NULL ) > mLastUpdateTime + 3 ) +    { +        const int num_squares = rand() % 20 + 4; +        int sqr1_r = rand() % 0x80 + 0x20; +        int sqr1_g = rand() % 0x80 + 0x20; +        int sqr1_b = rand() % 0x80 + 0x20; +        int sqr2_r = rand() % 0x80 + 0x20; +        int sqr2_g = rand() % 0x80 + 0x20; +        int sqr2_b = rand() % 0x80 + 0x20; + +        for ( int y1 = 0; y1 < num_squares; ++y1 ) +        { +            for ( int x1 = 0; x1 < num_squares; ++x1 ) +            { +                int px_start = mWidth * x1 / num_squares; +                int px_end = ( mWidth * ( x1 + 1 ) ) / num_squares; +                int py_start = mHeight * y1 / num_squares; +                int py_end = ( mHeight * ( y1 + 1 ) ) / num_squares; + +                for( int y2 = py_start; y2 < py_end; ++y2 ) +                { +                    for( int x2 = px_start; x2 < px_end; ++x2 ) +                    { +                        int rowspan = mWidth * mDepth; + +                        if ( ( y1 % 2 ) ^ ( x1 % 2 ) ) +                        { +                            mBackgroundPixels[ y2 * rowspan + x2 * mDepth + 0 ] = sqr1_r; +                            mBackgroundPixels[ y2 * rowspan + x2 * mDepth + 1 ] = sqr1_g; +                            mBackgroundPixels[ y2 * rowspan + x2 * mDepth + 2 ] = sqr1_b; +                        } +                        else +                        { +                            mBackgroundPixels[ y2 * rowspan + x2 * mDepth + 0 ] = sqr2_r; +                            mBackgroundPixels[ y2 * rowspan + x2 * mDepth + 1 ] = sqr2_g; +                            mBackgroundPixels[ y2 * rowspan + x2 * mDepth + 2 ] = sqr2_b; +                        }; +                    }; +                }; +            }; +        }; + +        time( &mLastUpdateTime ); +    }; + +    memcpy( mPixels, mBackgroundPixels, mWidth * mHeight * mDepth ); + +    for( int n = 0; n < ENumObjects; ++n ) +    { +        if ( rand() % 50 == 0 ) +        { +                mXInc[ n ] = 0; +                while ( mXInc[ n ] == 0 ) +                    mXInc[ n ] = rand() % 7 - 3; + +                mYInc[ n ] = 0; +                while ( mYInc[ n ] == 0 ) +                    mYInc[ n ] = rand() % 9 - 4; +        }; + +        if ( mXpos[ n ] + mXInc[ n ] < 0 || mXpos[ n ] + mXInc[ n ] >= mWidth - mBlockSize[ n ] ) +            mXInc[ n ] =- mXInc[ n ]; + +        if ( mYpos[ n ] + mYInc[ n ] < 0 || mYpos[ n ] + mYInc[ n ] >= mHeight - mBlockSize[ n ] ) +            mYInc[ n ] =- mYInc[ n ]; + +        mXpos[ n ] += mXInc[ n ]; +        mYpos[ n ] += mYInc[ n ]; + +        for( int y = 0; y < mBlockSize[ n ]; ++y ) +        { +            for( int x = 0; x < mBlockSize[ n ]; ++x ) +            { +                mPixels[ ( mXpos[ n ] + x ) * mDepth + ( mYpos[ n ] + y ) * mDepth * mWidth + 0 ] = mColorR[ n ]; +                mPixels[ ( mXpos[ n ] + x ) * mDepth + ( mYpos[ n ] + y ) * mDepth * mWidth + 1 ] = mColorG[ n ]; +                mPixels[ ( mXpos[ n ] + x ) * mDepth + ( mYpos[ n ] + y ) * mDepth * mWidth + 2 ] = mColorB[ n ]; +            }; +        }; +    }; + +    setDirty( 0, 0, mWidth, mHeight );  };  ////////////////////////////////////////////////////////////////////////////////  //  bool MediaPluginExample::init()  { -	LLPluginMessage message( LLPLUGIN_MESSAGE_CLASS_MEDIA, "name_text" ); -	message.setValue( "name", "Example Plugin" ); -	sendMessage( message ); +    LLPluginMessage message( LLPLUGIN_MESSAGE_CLASS_MEDIA, "name_text" ); +    message.setValue( "name", "Example Plugin" ); +    sendMessage( message ); -	return true; +    return true;  };  ////////////////////////////////////////////////////////////////////////////////  //  int init_media_plugin( LLPluginInstance::sendMessageFunction host_send_func, -						void* host_user_data, -						LLPluginInstance::sendMessageFunction *plugin_send_func, -						void **plugin_user_data ) +                        void* host_user_data, +                        LLPluginInstance::sendMessageFunction *plugin_send_func, +                        void **plugin_user_data )  { -	MediaPluginExample* self = new MediaPluginExample( host_send_func, host_user_data ); -	*plugin_send_func = MediaPluginExample::staticReceiveMessage; -	*plugin_user_data = ( void* )self; +    MediaPluginExample* self = new MediaPluginExample( host_send_func, host_user_data ); +    *plugin_send_func = MediaPluginExample::staticReceiveMessage; +    *plugin_user_data = ( void* )self; -	return 0; +    return 0;  } + diff --git a/indra/media_plugins/webkit/media_plugin_webkit.cpp b/indra/media_plugins/webkit/media_plugin_webkit.cpp index bd1a44a930..d6f8ae3e16 100644 --- a/indra/media_plugins/webkit/media_plugin_webkit.cpp +++ b/indra/media_plugins/webkit/media_plugin_webkit.cpp @@ -341,7 +341,7 @@ private:  		url << std::setfill('0') << std::setw(2) << std::hex << int(mBackgroundB * 255.0f);  		url << "%22%3E%3C/body%3E%3C/html%3E"; -		lldebugs << "data url is: " << url.str() << llendl; +		//lldebugs << "data url is: " << url.str() << llendl;  		LLQtWebKit::getInstance()->navigateTo( mBrowserWindowId, url.str() );  //		LLQtWebKit::getInstance()->navigateTo( mBrowserWindowId, "about:blank" ); @@ -407,6 +407,8 @@ private:  		{  			LLPluginMessage message(LLPLUGIN_MESSAGE_CLASS_MEDIA_BROWSER, "navigate_begin");  			message.setValue("uri", event.getEventUri()); +			message.setValueBoolean("history_back_available", LLQtWebKit::getInstance()->userActionIsEnabled( mBrowserWindowId, LLQtWebKit::UA_NAVIGATE_BACK)); +			message.setValueBoolean("history_forward_available", LLQtWebKit::getInstance()->userActionIsEnabled( mBrowserWindowId, LLQtWebKit::UA_NAVIGATE_FORWARD));  			sendMessage(message);  			setStatus(STATUS_LOADING); @@ -569,6 +571,57 @@ private:  		return blockingPickFile();  	} +	std::string mAuthUsername; +	std::string mAuthPassword; +	bool mAuthOK; +	 +	//////////////////////////////////////////////////////////////////////////////// +	// virtual +	bool onAuthRequest(const std::string &in_url, const std::string &in_realm, std::string &out_username, std::string &out_password) +	{ +		mAuthOK = false; + +		LLPluginMessage message(LLPLUGIN_MESSAGE_CLASS_MEDIA, "auth_request"); +		message.setValue("url", in_url); +		message.setValue("realm", in_realm); +		message.setValueBoolean("blocking_request", true); +				 +		// The "blocking_request" key in the message means this sendMessage call will block until a response is received. +		sendMessage(message); +		 +		if(mAuthOK) +		{ +			out_username = mAuthUsername; +			out_password = mAuthPassword; +		} +		 +		return mAuthOK; +	} +	 +	void authResponse(LLPluginMessage &message) +	{ +		mAuthOK = message.getValueBoolean("ok"); +		if(mAuthOK) +		{ +			mAuthUsername = message.getValue("username"); +			mAuthPassword = message.getValue("password"); +		} +	} +	 +	//////////////////////////////////////////////////////////////////////////////// +	// virtual +	void onLinkHovered(const EventType& event) +	{ +		if(mInitState >= INIT_STATE_NAVIGATE_COMPLETE) +		{ +			LLPluginMessage message(LLPLUGIN_MESSAGE_CLASS_MEDIA_BROWSER, "link_hovered"); +			message.setValue("link", event.getEventUri()); +			message.setValue("title", event.getStringValue()); +			message.setValue("text", event.getStringValue2()); +			sendMessage(message); +		} +	} +	  	LLQtWebKit::EKeyboardModifier decodeModifiers(std::string &modifiers)  	{  		int result = 0; @@ -1096,6 +1149,10 @@ void MediaPluginWebKit::receiveMessage(const char *message_string)  			{  				onPickFileResponse(message_in.getValue("file"));  			} +			if(message_name == "auth_response") +			{ +				authResponse(message_in); +			}  			else  			{  //				std::cerr << "MediaPluginWebKit::receiveMessage: unknown media message: " << message_string << std::endl; @@ -1182,6 +1239,22 @@ void MediaPluginWebKit::receiveMessage(const char *message_string)  				mUserAgent = message_in.getValue("user_agent");  				LLQtWebKit::getInstance()->setBrowserAgentId( mUserAgent );  			} +			else if(message_name == "ignore_ssl_cert_errors") +			{ +#if LLQTWEBKIT_API_VERSION >= 3 +				LLQtWebKit::getInstance()->setIgnoreSSLCertErrors( message_in.getValueBoolean("ignore") ); +#else +				llwarns << "Ignoring ignore_ssl_cert_errors message (llqtwebkit version is too old)." << llendl; +#endif +			} +			else if(message_name == "add_certificate_file_path") +			{ +#if LLQTWEBKIT_API_VERSION >= 6 +				LLQtWebKit::getInstance()->addCAFile( message_in.getValue("path") ); +#else +				llwarns << "Ignoring add_certificate_file_path message (llqtwebkit version is too old)." << llendl; +#endif +			}  			else if(message_name == "init_history")  			{  				// Initialize browser history diff --git a/indra/newview/CMakeLists.txt b/indra/newview/CMakeLists.txt index 4869aa72a0..7975b181d3 100644 --- a/indra/newview/CMakeLists.txt +++ b/indra/newview/CMakeLists.txt @@ -223,6 +223,7 @@ set(viewer_SOURCE_FILES      llfloaterurlentry.cpp      llfloatervoiceeffect.cpp      llfloaterwater.cpp +    llfloaterwebcontent.cpp      llfloaterwhitelistentry.cpp      llfloaterwindlight.cpp      llfloaterwindowsize.cpp @@ -759,6 +760,7 @@ set(viewer_HEADER_FILES      llfloaterurlentry.h      llfloatervoiceeffect.h      llfloaterwater.h +    llfloaterwebcontent.h      llfloaterwhitelistentry.h      llfloaterwindlight.h      llfloaterwindowsize.h diff --git a/indra/newview/app_settings/lindenlab.pem b/indra/newview/app_settings/lindenlab.pem new file mode 100644 index 0000000000..cf88d0e047 --- /dev/null +++ b/indra/newview/app_settings/lindenlab.pem @@ -0,0 +1,27 @@ +-----BEGIN CERTIFICATE----- +MIIEUDCCA7mgAwIBAgIJAN4ppNGwj6yIMA0GCSqGSIb3DQEBBAUAMIHMMQswCQYD +VQQGEwJVUzETMBEGA1UECBMKQ2FsaWZvcm5pYTEWMBQGA1UEBxMNU2FuIEZyYW5j +aXNjbzEZMBcGA1UEChMQTGluZGVuIExhYiwgSW5jLjEpMCcGA1UECxMgTGluZGVu +IExhYiBDZXJ0aWZpY2F0ZSBBdXRob3JpdHkxKTAnBgNVBAMTIExpbmRlbiBMYWIg +Q2VydGlmaWNhdGUgQXV0aG9yaXR5MR8wHQYJKoZIhvcNAQkBFhBjYUBsaW5kZW5s +YWIuY29tMB4XDTA1MDQyMTAyNDAzMVoXDTI1MDQxNjAyNDAzMVowgcwxCzAJBgNV +BAYTAlVTMRMwEQYDVQQIEwpDYWxpZm9ybmlhMRYwFAYDVQQHEw1TYW4gRnJhbmNp +c2NvMRkwFwYDVQQKExBMaW5kZW4gTGFiLCBJbmMuMSkwJwYDVQQLEyBMaW5kZW4g +TGFiIENlcnRpZmljYXRlIEF1dGhvcml0eTEpMCcGA1UEAxMgTGluZGVuIExhYiBD +ZXJ0aWZpY2F0ZSBBdXRob3JpdHkxHzAdBgkqhkiG9w0BCQEWEGNhQGxpbmRlbmxh +Yi5jb20wgZ8wDQYJKoZIhvcNAQEBBQADgY0AMIGJAoGBAKXh1MThucdTbMg9bYBO +rAm8yWns32YojB0PRfbq8rUjepEhTm3/13s0u399Uc202v4ejcGhkIDWJZd2NZMF +oKrhmRfxGHSKPCuFaXC3jh0lRECj7k8FoPkcmaPjSyodrDFDUUuv+C06oYJoI+rk +8REyal9NwgHvqCzOrZtiTXAdAgMBAAGjggE2MIIBMjAdBgNVHQ4EFgQUO1zK2e1f +1wO1fHAjq6DTJobKDrcwggEBBgNVHSMEgfkwgfaAFDtcytntX9cDtXxwI6ug0yaG +yg63oYHSpIHPMIHMMQswCQYDVQQGEwJVUzETMBEGA1UECBMKQ2FsaWZvcm5pYTEW +MBQGA1UEBxMNU2FuIEZyYW5jaXNjbzEZMBcGA1UEChMQTGluZGVuIExhYiwgSW5j +LjEpMCcGA1UECxMgTGluZGVuIExhYiBDZXJ0aWZpY2F0ZSBBdXRob3JpdHkxKTAn +BgNVBAMTIExpbmRlbiBMYWIgQ2VydGlmaWNhdGUgQXV0aG9yaXR5MR8wHQYJKoZI +hvcNAQkBFhBjYUBsaW5kZW5sYWIuY29tggkA3imk0bCPrIgwDAYDVR0TBAUwAwEB +/zANBgkqhkiG9w0BAQQFAAOBgQA/ZkgfvwHYqk1UIAKZS3kMCxz0HvYuEQtviwnu +xA39CIJ65Zozs28Eg1aV9/Y+Of7TnWhW+U3J3/wD/GghaAGiKK6vMn9gJBIdBX/9 +e6ef37VGyiOEFFjnUIbuk0RWty0orN76q/lI/xjCi15XSA/VSq2j4vmnwfZcPTDu +glmQ1A== +-----END CERTIFICATE----- + diff --git a/indra/newview/app_settings/settings.xml b/indra/newview/app_settings/settings.xml index 06992d2b52..199ee6822f 100644 --- a/indra/newview/app_settings/settings.xml +++ b/indra/newview/app_settings/settings.xml @@ -686,6 +686,39 @@        <key>Value</key>        <string>http://www.secondlife.com</string>      </map> +    <key>BrowserIgnoreSSLCertErrors</key> +    <map> +      <key>Comment</key> +      <string>FOR TESTING ONLY: Tell the built-in web browser to ignore SSL cert errors.</string> +      <key>Persist</key> +      <integer>1</integer> +      <key>Type</key> +      <string>Boolean</string> +      <key>Value</key> +      <integer>0</integer> +    </map> +    <key>BrowserUseDefaultCAFile</key> +    <map> +      <key>Comment</key> +      <string>Tell the built-in web browser to use the CA.pem file shipped with the client.</string> +      <key>Persist</key> +      <integer>1</integer> +      <key>Type</key> +      <string>Boolean</string> +      <key>Value</key> +      <integer>1</integer> +    </map> +    <key>BrowserCAFilePath</key> +    <map> +      <key>Comment</key> +      <string>Tell the built-in web browser the path to an alternative CA.pem file (only used if BrowserUseDefaultCAFile is false).</string> +      <key>Persist</key> +      <integer>1</integer> +      <key>Type</key> +      <string>String</string> +      <key>Value</key> +      <string></string> +    </map>        <key>BlockAvatarAppearanceMessages</key>          <map>          <key>Comment</key> @@ -6572,7 +6605,18 @@      <key>MediaBrowserWindowLimit</key>      <map>        <key>Comment</key> -      <string>Maximum number of media brower windows that can be open at once (0 for no limit)</string> +      <string>Maximum number of media brower windows that can be open at once in the media browser floater (0 for no limit)</string> +      <key>Persist</key> +      <integer>1</integer> +      <key>Type</key> +      <string>S32</string> +      <key>Value</key> +      <integer>5</integer> +    </map> +    <key>WebContentWindowLimit</key> +    <map> +      <key>Comment</key> +      <string>Maximum number of web brower windows that can be open at once in the Web content floater (0 for no limit)</string>        <key>Persist</key>        <integer>1</integer>        <key>Type</key> diff --git a/indra/newview/llbrowsernotification.cpp b/indra/newview/llbrowsernotification.cpp index d6a813d608..6e77d1e336 100644 --- a/indra/newview/llbrowsernotification.cpp +++ b/indra/newview/llbrowsernotification.cpp @@ -29,8 +29,9 @@  #include "llnotificationhandler.h"  #include "llnotifications.h" -#include "llfloaterreg.h"  #include "llmediactrl.h" +#include "llviewermedia.h" +#include "llviewermediafocus.h"  using namespace LLNotificationsUI; @@ -39,10 +40,19 @@ bool LLBrowserNotification::processNotification(const LLSD& notify)  	LLNotificationPtr notification = LLNotifications::instance().find(notify["id"].asUUID());  	if (!notification) return false; -	LLMediaCtrl* media_instance = LLMediaCtrl::getInstance(notification->getPayload()["media_id"].asUUID()); +	LLUUID media_id = notification->getPayload()["media_id"].asUUID(); +	LLMediaCtrl* media_instance = LLMediaCtrl::getInstance(media_id);  	if (media_instance)  	{  		media_instance->showNotification(notification);  	} +	else if (LLViewerMediaFocus::instance().getControlsMediaID() == media_id) +	{ +		LLViewerMediaImpl* impl = LLViewerMedia::getMediaImplFromTextureID(media_id); +		if (impl) +		{ +			impl->showNotification(notification); +		} +	}  	return false;  } diff --git a/indra/newview/llchathistory.cpp b/indra/newview/llchathistory.cpp index 6e778de2d8..c98bcbda45 100644 --- a/indra/newview/llchathistory.cpp +++ b/indra/newview/llchathistory.cpp @@ -586,7 +586,7 @@ void LLChatHistory::initFromParams(const LLChatHistory::Params& p)  	LLLayoutStack::Params layout_p;  	layout_p.rect = stack_rect;  	layout_p.follows.flags = FOLLOWS_ALL; -	layout_p.orientation = "vertical"; +	layout_p.orientation = LLLayoutStack::VERTICAL;  	layout_p.mouse_opaque = false;  	LLLayoutStack* stackp = LLUICtrlFactory::create<LLLayoutStack>(layout_p, this); diff --git a/indra/newview/llfloaterabout.cpp b/indra/newview/llfloaterabout.cpp index 8ae3ccbae3..2873bc0059 100644 --- a/indra/newview/llfloaterabout.cpp +++ b/indra/newview/llfloaterabout.cpp @@ -272,7 +272,7 @@ LLSD LLFloaterAbout::getInfo()  	}  	// TODO: Implement media plugin version query -	info["QT_WEBKIT_VERSION"] = "4.6 (version number hard-coded)"; +	info["QT_WEBKIT_VERSION"] = "4.7.1 (version number hard-coded)";  	if (gPacketsIn > 0)  	{ diff --git a/indra/newview/llfloaterhelpbrowser.cpp b/indra/newview/llfloaterhelpbrowser.cpp index cec98e9992..a650886d89 100644 --- a/indra/newview/llfloaterhelpbrowser.cpp +++ b/indra/newview/llfloaterhelpbrowser.cpp @@ -132,9 +132,10 @@ void LLFloaterHelpBrowser::onClickOpenWebBrowser(void* user_data)  void LLFloaterHelpBrowser::openMedia(const std::string& media_url)  { -	mBrowser->setHomePageUrl(media_url); -	//mBrowser->navigateTo("data:text/html;charset=utf-8,I'd really love to be going to:<br><b>" + media_url + "</b>"); // tofu HACK for debugging =:) -	mBrowser->navigateTo(media_url); +	// explicitly make the media mime type for this floater since it will +	// only ever display one type of content (Web). +	mBrowser->setHomePageUrl(media_url, "text/html"); +	mBrowser->navigateTo(media_url, "text/html");  	setCurrentURL(media_url);  } diff --git a/indra/newview/llfloatermediabrowser.cpp b/indra/newview/llfloatermediabrowser.cpp index d20092e344..7a670dd90c 100644 --- a/indra/newview/llfloatermediabrowser.cpp +++ b/indra/newview/llfloatermediabrowser.cpp @@ -306,17 +306,14 @@ void LLFloaterMediaBrowser::setCurrentURL(const std::string& url)  {  	mCurrentURL = url; -	// redirects will navigate momentarily to about:blank, don't add to history -	if (mCurrentURL != "about:blank") -	{ -		mAddressCombo->remove(mCurrentURL); -		mAddressCombo->add(mCurrentURL); -		mAddressCombo->selectByValue(mCurrentURL); +	mAddressCombo->remove(mCurrentURL); +	mAddressCombo->add(mCurrentURL); +	mAddressCombo->selectByValue(mCurrentURL); + +	// Serialize url history +	LLURLHistory::removeURL("browser", mCurrentURL); +	LLURLHistory::addURL("browser", mCurrentURL); -		// Serialize url history -		LLURLHistory::removeURL("browser", mCurrentURL); -		LLURLHistory::addURL("browser", mCurrentURL); -	}  	getChildView("back")->setEnabled(mBrowser->canNavigateBack());  	getChildView("forward")->setEnabled(mBrowser->canNavigateForward());  	getChildView("reload")->setEnabled(TRUE); @@ -334,8 +331,15 @@ void LLFloaterMediaBrowser::onClickRefresh(void* user_data)  {  	LLFloaterMediaBrowser* self = (LLFloaterMediaBrowser*)user_data; -	self->mAddressCombo->remove(0); -	self->mBrowser->navigateTo(self->mCurrentURL); +	if( self->mBrowser->getMediaPlugin() &&  self->mBrowser->getMediaPlugin()->pluginSupportsMediaBrowser()) +	{ +		bool ignore_cache = true; +		self->mBrowser->getMediaPlugin()->browse_reload( ignore_cache ); +	} +	else +	{ +		self->mBrowser->navigateTo(self->mCurrentURL); +	}  }  //static  diff --git a/indra/newview/llfloatersearch.cpp b/indra/newview/llfloatersearch.cpp index 3ed4aec89a..2041fac8d8 100644 --- a/indra/newview/llfloatersearch.cpp +++ b/indra/newview/llfloatersearch.cpp @@ -200,5 +200,5 @@ void LLFloaterSearch::search(const LLSD &key)  	url = LLWeb::expandURLSubstitutions(url, subs);  	// and load the URL in the web view -	mBrowser->navigateTo(url); +	mBrowser->navigateTo(url, "text/html");  } diff --git a/indra/newview/llfloaterwebcontent.cpp b/indra/newview/llfloaterwebcontent.cpp new file mode 100644 index 0000000000..51726112a0 --- /dev/null +++ b/indra/newview/llfloaterwebcontent.cpp @@ -0,0 +1,402 @@ +/**
 + * @file llfloaterwebcontent.cpp
 + * @brief floater for displaying web content - e.g. profiles and search (eventually)
 + *
 + * $LicenseInfo:firstyear=2006&license=viewerlgpl$
 + * Second Life Viewer Source Code
 + * Copyright (C) 2010, Linden Research, Inc.
 + *
 + * This library is free software; you can redistribute it and/or
 + * modify it under the terms of the GNU Lesser General Public
 + * License as published by the Free Software Foundation;
 + * version 2.1 of the License only.
 + *
 + * This library is distributed in the hope that it will be useful,
 + * but WITHOUT ANY WARRANTY; without even the implied warranty of
 + * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU
 + * Lesser General Public License for more details.
 + *
 + * You should have received a copy of the GNU Lesser General Public
 + * License along with this library; if not, write to the Free Software
 + * Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA  02110-1301  USA
 + *
 + * Linden Research, Inc., 945 Battery Street, San Francisco, CA  94111  USA
 + * $/LicenseInfo$
 + */
 +
 +#include "llviewerprecompiledheaders.h"
 +
 +#include "llcombobox.h"
 +#include "lliconctrl.h"
 +#include "llfloaterreg.h"
 +#include "lllayoutstack.h"
 +#include "llpluginclassmedia.h"
 +#include "llprogressbar.h"
 +#include "lltextbox.h"
 +#include "llurlhistory.h"
 +#include "llviewercontrol.h"
 +#include "llweb.h"
 +#include "llwindow.h"
 +
 +#include "llfloaterwebcontent.h"
 +
 +LLFloaterWebContent::LLFloaterWebContent( const LLSD& key )
 +	: LLFloater( key )
 +{
 +	mCommitCallbackRegistrar.add( "WebContent.Back", boost::bind( &LLFloaterWebContent::onClickBack, this ));
 +	mCommitCallbackRegistrar.add( "WebContent.Forward", boost::bind( &LLFloaterWebContent::onClickForward, this ));
 +	mCommitCallbackRegistrar.add( "WebContent.Reload", boost::bind( &LLFloaterWebContent::onClickReload, this ));
 +	mCommitCallbackRegistrar.add( "WebContent.Stop", boost::bind( &LLFloaterWebContent::onClickStop, this ));
 +	mCommitCallbackRegistrar.add( "WebContent.EnterAddress", boost::bind( &LLFloaterWebContent::onEnterAddress, this ));
 +	mCommitCallbackRegistrar.add( "WebContent.PopExternal", boost::bind( &LLFloaterWebContent::onPopExternal, this ));
 +}
 +
 +BOOL LLFloaterWebContent::postBuild()
 +{
 +	// these are used in a bunch of places so cache them
 +	mWebBrowser = getChild< LLMediaCtrl >( "webbrowser" );
 +	mAddressCombo = getChild< LLComboBox >( "address" );
 +	mStatusBarText = getChild< LLTextBox >( "statusbartext" );
 +	mStatusBarProgress = getChild<LLProgressBar>("statusbarprogress" );
 +
 +	// observe browser events
 +	mWebBrowser->addObserver( this );
 +
 +	// these buttons are always enabled
 +	getChildView("reload")->setEnabled( true );
 +	getChildView("popexternal")->setEnabled( true );
 +
 +	// cache image for secure browsing
 +	mSecureLockIcon = getChild< LLIconCtrl >("media_secure_lock_flag");
 +
 +	// initialize the URL history using the system URL History manager
 +	initializeURLHistory();
 +
 +	return TRUE;
 +}
 +
 +void LLFloaterWebContent::initializeURLHistory()
 +{
 +	// start with an empty list
 +	LLCtrlListInterface* url_list = childGetListInterface("address");
 +	if (url_list)
 +	{
 +		url_list->operateOnAll(LLCtrlListInterface::OP_DELETE);
 +	}
 +
 +	// Get all of the entries in the "browser" collection
 +	LLSD browser_history = LLURLHistory::getURLHistory("browser");
 +	LLSD::array_iterator iter_history =
 +		browser_history.beginArray();
 +	LLSD::array_iterator end_history =
 +		browser_history.endArray();
 +	for(; iter_history != end_history; ++iter_history)
 +	{
 +		std::string url = (*iter_history).asString();
 +		if(! url.empty())
 +			url_list->addSimpleElement(url);
 +	}
 +}
 +
 +//static
 +void LLFloaterWebContent::create( const std::string &url, const std::string& target, const std::string& uuid )
 +{
 +	lldebugs << "url = " << url << ", target = " << target << ", uuid = " << uuid << llendl;
 +
 +	std::string tag = target;
 +
 +	if(target.empty() || target == "_blank")
 +	{
 +		if(!uuid.empty())
 +		{
 +			tag = uuid;
 +		}
 +		else
 +		{
 +			// create a unique tag for this instance
 +			LLUUID id;
 +			id.generate();
 +			tag = id.asString();
 +		}
 +	}
 +
 +	S32 browser_window_limit = gSavedSettings.getS32("WebContentWindowLimit");
 +
 +	if(LLFloaterReg::findInstance("web_content", tag) != NULL)
 +	{
 +		// There's already a web browser for this tag, so we won't be opening a new window.
 +	}
 +	else if(browser_window_limit != 0)
 +	{
 +		// showInstance will open a new window.  Figure out how many web browsers are already open,
 +		// and close the least recently opened one if this will put us over the limit.
 +
 +		LLFloaterReg::const_instance_list_t &instances = LLFloaterReg::getFloaterList("web_content");
 +		lldebugs << "total instance count is " << instances.size() << llendl;
 +
 +		for(LLFloaterReg::const_instance_list_t::const_iterator iter = instances.begin(); iter != instances.end(); iter++)
 +		{
 +			lldebugs << "    " << (*iter)->getKey() << llendl;
 +		}
 +
 +		if(instances.size() >= (size_t)browser_window_limit)
 +		{
 +			// Destroy the least recently opened instance
 +			(*instances.begin())->closeFloater();
 +		}
 +	}
 +
 +	LLFloaterWebContent *browser = dynamic_cast<LLFloaterWebContent*> (LLFloaterReg::showInstance("web_content", tag));
 +	llassert(browser);
 +	if(browser)
 +	{
 +		browser->mUUID = uuid;
 +
 +		// tell the browser instance to load the specified URL
 +		browser->open_media(url, target);
 +		LLViewerMedia::proxyWindowOpened(target, uuid);
 +	}
 +}
 +
 +//static
 +void LLFloaterWebContent::closeRequest(const std::string &uuid)
 +{
 +	LLFloaterReg::const_instance_list_t& inst_list = LLFloaterReg::getFloaterList("web_content");
 +	lldebugs << "instance list size is " << inst_list.size() << ", incoming uuid is " << uuid << llendl;
 +	for (LLFloaterReg::const_instance_list_t::const_iterator iter = inst_list.begin(); iter != inst_list.end(); ++iter)
 +	{
 +		LLFloaterWebContent* i = dynamic_cast<LLFloaterWebContent*>(*iter);
 +		lldebugs << "    " << i->mUUID << llendl;
 +		if (i && i->mUUID == uuid)
 +		{
 +			i->closeFloater(false);
 +			return;
 + 		}
 + 	}
 +}
 +
 +//static
 +void LLFloaterWebContent::geometryChanged(const std::string &uuid, S32 x, S32 y, S32 width, S32 height)
 +{
 +	LLFloaterReg::const_instance_list_t& inst_list = LLFloaterReg::getFloaterList("web_content");
 +	lldebugs << "instance list size is " << inst_list.size() << ", incoming uuid is " << uuid << llendl;
 +	for (LLFloaterReg::const_instance_list_t::const_iterator iter = inst_list.begin(); iter != inst_list.end(); ++iter)
 +	{
 +		LLFloaterWebContent* i = dynamic_cast<LLFloaterWebContent*>(*iter);
 +		lldebugs << "    " << i->mUUID << llendl;
 +		if (i && i->mUUID == uuid)
 +		{
 +			i->geometryChanged(x, y, width, height);
 +			return;
 +		}
 +	}
 +}
 +
 +void LLFloaterWebContent::geometryChanged(S32 x, S32 y, S32 width, S32 height)
 +{
 +	// Make sure the layout of the browser control is updated, so this calculation is correct.
 +	LLLayoutStack::updateClass();
 +
 +	// TODO: need to adjust size and constrain position to make sure floaters aren't moved outside the window view, etc.
 +	LLCoordWindow window_size;
 +	getWindow()->getSize(&window_size);
 +
 +	// Adjust width and height for the size of the chrome on the web Browser window.
 +	width += getRect().getWidth() - mWebBrowser->getRect().getWidth();
 +	height += getRect().getHeight() - mWebBrowser->getRect().getHeight();
 +
 +	LLRect geom;
 +	geom.setOriginAndSize(x, window_size.mY - (y + height), width, height);
 +
 +	lldebugs << "geometry change: " << geom << llendl;
 +
 +	handleReshape(geom,false);
 +}
 +
 +void LLFloaterWebContent::open_media(const std::string& web_url, const std::string& target)
 +{
 +	// Specifying a mime type of text/html here causes the plugin system to skip the MIME type probe and just open a browser plugin.
 +	mWebBrowser->setHomePageUrl(web_url, "text/html");
 +	mWebBrowser->setTarget(target);
 +	mWebBrowser->navigateTo(web_url, "text/html");
 +	set_current_url(web_url);
 +}
 +
 +//virtual
 +void LLFloaterWebContent::onClose(bool app_quitting)
 +{
 +	LLViewerMedia::proxyWindowClosed(mUUID);
 +	destroy();
 +}
 +
 +// virtual
 +void LLFloaterWebContent::draw()
 +{
 +	// this is asychronous so we need to keep checking
 +	getChildView( "back" )->setEnabled( mWebBrowser->canNavigateBack() );
 +	getChildView( "forward" )->setEnabled( mWebBrowser->canNavigateForward() );
 +
 +	LLFloater::draw();
 +}
 +
 +// virtual
 +void LLFloaterWebContent::handleMediaEvent(LLPluginClassMedia* self, EMediaEvent event)
 +{
 +	if(event == MEDIA_EVENT_LOCATION_CHANGED)
 +	{
 +		const std::string url = self->getLocation();
 +
 +		if ( url.length() )
 +			mStatusBarText->setText( url );
 +
 +		set_current_url( url );
 +	}
 +	else if(event == MEDIA_EVENT_NAVIGATE_BEGIN)
 +	{
 +		// flags are sent with this event
 +		getChildView("back")->setEnabled( self->getHistoryBackAvailable() );
 +		getChildView("forward")->setEnabled( self->getHistoryForwardAvailable() );
 +
 +		// toggle visibility of these buttons based on browser state
 +		getChildView("reload")->setVisible( false );
 +		getChildView("stop")->setVisible( true );
 +
 +		// turn "on" progress bar now we're about to start loading
 +		mStatusBarProgress->setVisible( true );
 +	}
 +	else if(event == MEDIA_EVENT_NAVIGATE_COMPLETE)
 +	{
 +		// flags are sent with this event
 +		getChildView("back")->setEnabled( self->getHistoryBackAvailable() );
 +		getChildView("forward")->setEnabled( self->getHistoryForwardAvailable() );
 +
 +		// toggle visibility of these buttons based on browser state
 +		getChildView("reload")->setVisible( true );
 +		getChildView("stop")->setVisible( false );
 +
 +		// turn "off" progress bar now we're loaded
 +		mStatusBarProgress->setVisible( false );
 +
 +		// we populate the status bar with URLs as they change so clear it now we're done
 +		const std::string end_str = "";
 +		mStatusBarText->setText( end_str );
 +
 +		// decide if secure browsing icon should be displayed
 +		std::string prefix =  std::string("https://");
 +		std::string test_prefix = mCurrentURL.substr(0, prefix.length());
 +		LLStringUtil::toLower(test_prefix);
 +		if(test_prefix == prefix)
 +		{
 +			mSecureLockIcon->setVisible(true);
 +		}
 +		else
 +		{
 +			mSecureLockIcon->setVisible(false);
 +		}
 +	}
 +	else if(event == MEDIA_EVENT_CLOSE_REQUEST)
 +	{
 +		// The browser instance wants its window closed.
 +		closeFloater();
 +	}
 +	else if(event == MEDIA_EVENT_GEOMETRY_CHANGE)
 +	{
 +		geometryChanged(self->getGeometryX(), self->getGeometryY(), self->getGeometryWidth(), self->getGeometryHeight());
 +	}
 +	else if(event == MEDIA_EVENT_STATUS_TEXT_CHANGED )
 +	{
 +		const std::string text = self->getStatusText();
 +		if ( text.length() )
 +			mStatusBarText->setText( text );
 +	}
 +	else if(event == MEDIA_EVENT_PROGRESS_UPDATED )
 +	{
 +		int percent = (int)self->getProgressPercent();
 +		mStatusBarProgress->setValue( percent );
 +	}
 +	else if(event == MEDIA_EVENT_NAME_CHANGED )
 +	{
 +		std::string page_title = self->getMediaName();
 +		// simulate browser behavior - title is empty, use the current URL
 +		if ( page_title.length() > 0 )
 +			setTitle( page_title );
 +		else
 +			setTitle( mCurrentURL );
 +	}
 +	else if(event == MEDIA_EVENT_LINK_HOVERED )
 +	{
 +		const std::string link = self->getHoverLink();
 +		mStatusBarText->setText( link );
 +	}
 +}
 +
 +void LLFloaterWebContent::set_current_url(const std::string& url)
 +{
 +	mCurrentURL = url;
 +
 +	// serialize url history into the system URL History manager
 +	LLURLHistory::removeURL("browser", mCurrentURL);
 +	LLURLHistory::addURL("browser", mCurrentURL);
 +
 +	mAddressCombo->remove( mCurrentURL );
 +	mAddressCombo->add( mCurrentURL );
 +	mAddressCombo->selectByValue( mCurrentURL );
 +}
 +
 +void LLFloaterWebContent::onClickForward()
 +{
 +	mWebBrowser->navigateForward();
 +}
 +
 +void LLFloaterWebContent::onClickBack()
 +{
 +	mWebBrowser->navigateBack();
 +}
 +
 +void LLFloaterWebContent::onClickReload()
 +{
 +
 +	if( mWebBrowser->getMediaPlugin() )
 +	{
 +		bool ignore_cache = true;
 +		mWebBrowser->getMediaPlugin()->browse_reload( ignore_cache );
 +	}
 +	else
 +	{
 +		mWebBrowser->navigateTo(mCurrentURL);
 +	}
 +}
 +
 +void LLFloaterWebContent::onClickStop()
 +{
 +	if( mWebBrowser->getMediaPlugin() )
 +		mWebBrowser->getMediaPlugin()->browse_stop();
 +
 +	// still should happen when we catch the navigate complete event
 +	// but sometimes (don't know why) that event isn't sent from Qt
 +	// and we getto a point where the stop button stays active.
 +	getChildView("reload")->setVisible( true );
 +	getChildView("stop")->setVisible( false );
 +}
 +
 +void LLFloaterWebContent::onEnterAddress()
 +{
 +	// make sure there is at least something there.
 +	// (perhaps this test should be for minimum length of a URL)
 +	std::string url = mAddressCombo->getValue().asString();
 +	if ( url.length() > 0 )
 +	{
 +		mWebBrowser->navigateTo( url, "text/html");
 +	};
 +}
 +
 +void LLFloaterWebContent::onPopExternal()
 +{
 +	// make sure there is at least something there.
 +	// (perhaps this test should be for minimum length of a URL)
 +	std::string url = mAddressCombo->getValue().asString();
 +	if ( url.length() > 0 )
 +	{
 +		LLWeb::loadURLExternal( url );
 +	};
 +}
 diff --git a/indra/newview/llfloaterwebcontent.h b/indra/newview/llfloaterwebcontent.h new file mode 100644 index 0000000000..001d822ada --- /dev/null +++ b/indra/newview/llfloaterwebcontent.h @@ -0,0 +1,82 @@ +/**
 + * @file llfloaterwebcontent.h
 + * @brief floater for displaying web content - e.g. profiles and search (eventually)
 + *
 + * $LicenseInfo:firstyear=2006&license=viewerlgpl$
 + * Second Life Viewer Source Code
 + * Copyright (C) 2010, Linden Research, Inc.
 + *
 + * This library is free software; you can redistribute it and/or
 + * modify it under the terms of the GNU Lesser General Public
 + * License as published by the Free Software Foundation;
 + * version 2.1 of the License only.
 + *
 + * This library is distributed in the hope that it will be useful,
 + * but WITHOUT ANY WARRANTY; without even the implied warranty of
 + * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU
 + * Lesser General Public License for more details.
 + *
 + * You should have received a copy of the GNU Lesser General Public
 + * License along with this library; if not, write to the Free Software
 + * Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA  02110-1301  USA
 + *
 + * Linden Research, Inc., 945 Battery Street, San Francisco, CA  94111  USA
 + * $/LicenseInfo$
 + */
 +
 +#ifndef LL_LLFLOATERWEBCONTENT_H
 +#define LL_LLFLOATERWEBCONTENT_H
 +
 +#include "llfloater.h"
 +#include "llmediactrl.h"
 +
 +class LLMediaCtrl;
 +class LLComboBox;
 +class LLTextBox;
 +class LLProgressBar;
 +class LLIconCtrl;
 +
 +class LLFloaterWebContent :
 +	public LLFloater,
 +	public LLViewerMediaObserver
 +{
 +public:
 +    LOG_CLASS(LLFloaterWebContent);
 +	LLFloaterWebContent(const LLSD& key);
 +
 +	void initializeURLHistory(); +
 +	static void create(const std::string &url, const std::string& target, const std::string& uuid = LLStringUtil::null);
 +
 +	static void closeRequest(const std::string &uuid);
 +	static void geometryChanged(const std::string &uuid, S32 x, S32 y, S32 width, S32 height);
 +	void geometryChanged(S32 x, S32 y, S32 width, S32 height);
 +
 +	/* virtual */ BOOL postBuild();
 +	/* virtual */ void onClose(bool app_quitting);
 +	/* virtual */ void draw();
 +
 +	// inherited from LLViewerMediaObserver
 +	/*virtual*/ void handleMediaEvent(LLPluginClassMedia* self, EMediaEvent event);
 +
 +	void onClickBack();
 +	void onClickForward();
 +	void onClickReload();
 +	void onClickStop();
 +	void onEnterAddress();
 +	void onPopExternal();
 +
 +private:
 +	void open_media(const std::string& media_url, const std::string& target);
 +	void set_current_url(const std::string& url);
 +
 +	LLMediaCtrl* mWebBrowser;
 +	LLComboBox* mAddressCombo;
 +	LLIconCtrl *mSecureLockIcon;
 +	LLTextBox* mStatusBarText;
 +	LLProgressBar* mStatusBarProgress;
 +	std::string mCurrentURL;
 +	std::string mUUID;
 +};
 +
 +#endif  // LL_LLFLOATERWEBCONTENT_H
 diff --git a/indra/newview/llmediactrl.cpp b/indra/newview/llmediactrl.cpp index 0f66713ab0..9493fddf50 100644 --- a/indra/newview/llmediactrl.cpp +++ b/indra/newview/llmediactrl.cpp @@ -25,7 +25,7 @@   */  #include "llviewerprecompiledheaders.h" - +#include "lltooltip.h"  #include "llmediactrl.h" @@ -54,6 +54,10 @@  #include "llbutton.h"  #include "llcheckboxctrl.h"  #include "llnotifications.h" +#include "lllineeditor.h" +#include "llfloatermediabrowser.h" +#include "llfloaterwebcontent.h" +#include "llwindowshade.h"  extern BOOL gRestoreGL; @@ -98,7 +102,9 @@ LLMediaCtrl::LLMediaCtrl( const Params& p) :  	mTextureHeight ( 1024 ),  	mClearCache(false),  	mHomePageMimeType(p.initial_mime_type), -	mTrusted(p.trusted_content) +	mTrusted(p.trusted_content), +	mWindowShade(NULL), +	mHoverTextChanged(false)  {  	{  		LLColor4 color = p.caret_color().get(); @@ -127,7 +133,7 @@ LLMediaCtrl::LLMediaCtrl( const Params& p) :  		setTextureSize(screen_width, screen_height);  	} -	mMediaTextureID.generate(); +	mMediaTextureID = getKey();  	// We don't need to create the media source up front anymore unless we have a non-empty home URL to navigate to.  	if(!mHomePageUrl.empty()) @@ -141,8 +147,6 @@ LLMediaCtrl::LLMediaCtrl( const Params& p) :  //	addChild( mBorder );  } -//////////////////////////////////////////////////////////////////////////////// -// note: this is now a singleton and destruction happens via initClass() now  LLMediaCtrl::~LLMediaCtrl()  { @@ -182,6 +186,13 @@ BOOL LLMediaCtrl::handleHover( S32 x, S32 y, MASK mask )  		mMediaSource->mouseMove(x, y, mask);  		gViewerWindow->setCursor(mMediaSource->getLastSetCursor());  	} +	 +	// TODO: Is this the right way to handle hover text changes driven by the plugin? +	if(mHoverTextChanged) +	{ +		mHoverTextChanged = false; +		handleToolTip(x, y, mask); +	}  	return TRUE;  } @@ -198,6 +209,35 @@ BOOL LLMediaCtrl::handleScrollWheel( S32 x, S32 y, S32 clicks )  }  //////////////////////////////////////////////////////////////////////////////// +//	virtual  +BOOL LLMediaCtrl::handleToolTip(S32 x, S32 y, MASK mask) +{ +	std::string hover_text; +	 +	if (mMediaSource && mMediaSource->hasMedia()) +		hover_text = mMediaSource->getMediaPlugin()->getHoverText(); +	 +	if(hover_text.empty()) +	{ +		return FALSE; +	} +	else +	{ +		S32 screen_x, screen_y; + +		localPointToScreen(x, y, &screen_x, &screen_y); +		LLRect sticky_rect_screen; +		sticky_rect_screen.setCenterAndSize(screen_x, screen_y, 20, 20); + +		LLToolTipMgr::instance().show(LLToolTip::Params() +			.message(hover_text) +			.sticky_rect(sticky_rect_screen));		 +	} + +	return TRUE; +} + +////////////////////////////////////////////////////////////////////////////////  //  BOOL LLMediaCtrl::handleMouseUp( S32 x, S32 y, MASK mask )  { @@ -338,85 +378,6 @@ void LLMediaCtrl::onFocusLost()  //  BOOL LLMediaCtrl::postBuild ()  { -	LLLayoutStack::Params layout_p; -	layout_p.name = "notification_stack"; -	layout_p.rect = LLRect(0,getLocalRect().mTop,getLocalRect().mRight, 30); -	layout_p.follows.flags = FOLLOWS_ALL; -	layout_p.mouse_opaque = false; -	layout_p.orientation = "vertical"; - -	LLLayoutStack* stackp = LLUICtrlFactory::create<LLLayoutStack>(layout_p); -	addChild(stackp); - -	LLLayoutPanel::Params panel_p; -	panel_p.rect = LLRect(0, 30, 800, 0); -	panel_p.min_height = 30; -	panel_p.name = "notification_area"; -	panel_p.visible = false; -	panel_p.user_resize = false; -	panel_p.background_visible = true; -	panel_p.bg_alpha_image.name = "Yellow_Gradient"; -	panel_p.auto_resize = false; -	LLLayoutPanel* notification_panel = LLUICtrlFactory::create<LLLayoutPanel>(panel_p); -	stackp->addChild(notification_panel); - -	panel_p = LLUICtrlFactory::getDefaultParams<LLLayoutPanel>(); -	panel_p.auto_resize = true; -	panel_p.mouse_opaque = false; -	LLLayoutPanel* dummy_panel = LLUICtrlFactory::create<LLLayoutPanel>(panel_p); -	stackp->addChild(dummy_panel); - -	layout_p = LLUICtrlFactory::getDefaultParams<LLLayoutStack>(); -	layout_p.rect = LLRect(0, 30, 800, 0); -	layout_p.follows.flags = FOLLOWS_ALL; -	layout_p.orientation = "horizontal"; -	stackp = LLUICtrlFactory::create<LLLayoutStack>(layout_p); -	notification_panel->addChild(stackp); - -	panel_p = LLUICtrlFactory::getDefaultParams<LLLayoutPanel>(); -	panel_p.rect.height = 30; -	LLLayoutPanel* panel = LLUICtrlFactory::create<LLLayoutPanel>(panel_p); -	stackp->addChild(panel); - -	LLIconCtrl::Params icon_p; -	icon_p.name = "notification_icon"; -	icon_p.rect = LLRect(5, 23, 21, 8); -	panel->addChild(LLUICtrlFactory::create<LLIconCtrl>(icon_p)); - -	LLTextBox::Params text_p; -	text_p.rect = LLRect(31, 20, 430, 0); -	text_p.text_color = LLColor4::black; -	text_p.font = LLFontGL::getFontSansSerif(); -	text_p.font.style = "BOLD"; -	text_p.name = "notification_text"; -	text_p.use_ellipses = true; -	panel->addChild(LLUICtrlFactory::create<LLTextBox>(text_p)); - -	panel_p = LLUICtrlFactory::getDefaultParams<LLLayoutPanel>(); -	panel_p.auto_resize = false; -	panel_p.user_resize = false; -	panel_p.name="form_elements"; -	panel_p.rect = LLRect(0, 30, 130, 0); -	LLLayoutPanel* form_elements_panel = LLUICtrlFactory::create<LLLayoutPanel>(panel_p); -	stackp->addChild(form_elements_panel); - -	panel_p = LLUICtrlFactory::getDefaultParams<LLLayoutPanel>(); -	panel_p.auto_resize = false; -	panel_p.user_resize = false; -	panel_p.rect = LLRect(0, 30, 25, 0); -	LLLayoutPanel* close_panel = LLUICtrlFactory::create<LLLayoutPanel>(panel_p); -	stackp->addChild(close_panel); - -	LLButton::Params button_p; -	button_p.name = "close_notification"; -	button_p.rect = LLRect(5, 23, 21, 7); -	button_p.image_color=LLUIColorTable::instance().getColor("DkGray_66"); -    button_p.image_unselected.name="Icon_Close_Foreground"; -	button_p.image_selected.name="Icon_Close_Press"; -	button_p.click_callback.function = boost::bind(&LLMediaCtrl::onCloseNotification, this); - -	close_panel->addChild(LLUICtrlFactory::create<LLButton>(button_p)); -  	setVisibleCallback(boost::bind(&LLMediaCtrl::onVisibilityChange, this, _2));  	return TRUE;  } @@ -425,13 +386,15 @@ BOOL LLMediaCtrl::postBuild ()  //  BOOL LLMediaCtrl::handleKeyHere( KEY key, MASK mask )  { -	if (LLPanel::handleKeyHere(key, mask)) return TRUE;  	BOOL result = FALSE;  	if (mMediaSource)  	{  		result = mMediaSource->handleKeyHere(key, mask);  	} +	 +	if ( ! result ) +		result = LLPanel::handleKeyHere(key, mask);  	return result;  } @@ -451,7 +414,6 @@ void LLMediaCtrl::handleVisibilityChange ( BOOL new_visibility )  //  BOOL LLMediaCtrl::handleUnicodeCharHere(llwchar uni_char)  { -	if (LLPanel::handleUnicodeCharHere(uni_char)) return TRUE;  	BOOL result = FALSE;  	if (mMediaSource) @@ -459,6 +421,9 @@ BOOL LLMediaCtrl::handleUnicodeCharHere(llwchar uni_char)  		result = mMediaSource->handleUnicodeCharHere(uni_char);  	} +	if ( ! result ) +		result = LLPanel::handleUnicodeCharHere(uni_char); +  	return result;  } @@ -914,11 +879,6 @@ void LLMediaCtrl::draw()  	if ( mBorder && mBorder->getVisible() )  		mBorder->setKeyboardFocusHighlight( gFocusMgr.childHasKeyboardFocus( this ) ); -	if (mCurNotification && !mCurNotification->isActive()) -	{ -		hideNotification(); -	} -	  	LLPanel::draw();  	// Restore the previous values @@ -1026,7 +986,7 @@ void LLMediaCtrl::handleMediaEvent(LLPluginClassMedia* self, EMediaEvent event)  			LLNotification::Params notify_params;  			notify_params.name = "PopupAttempt"; -			notify_params.payload = LLSD().with("target", target).with("url", url).with("uuid", uuid).with("media_id", getKey()); +			notify_params.payload = LLSD().with("target", target).with("url", url).with("uuid", uuid).with("media_id", mMediaTextureID);  			notify_params.functor.function = boost::bind(&LLMediaCtrl::onPopup, this, _1, _2);  			if (mTrusted) @@ -1081,6 +1041,31 @@ void LLMediaCtrl::handleMediaEvent(LLPluginClassMedia* self, EMediaEvent event)  			LL_DEBUGS("Media") << "Media event:  MEDIA_EVENT_GEOMETRY_CHANGE, uuid is " << self->getClickUUID() << LL_ENDL;  		}  		break; + +		case MEDIA_EVENT_AUTH_REQUEST: +		{ +			LLNotification::Params auth_request_params; +			auth_request_params.name = "AuthRequest"; + +			// pass in host name and realm for site (may be zero length but will always exist) +			LLSD args; +			LLURL raw_url( self->getAuthURL().c_str() ); +			args["HOST_NAME"] = raw_url.getAuthority(); +			args["REALM"] = self->getAuthRealm(); +			auth_request_params.substitutions = args; + +			auth_request_params.payload = LLSD().with("media_id", mMediaTextureID); +			auth_request_params.functor.function = boost::bind(&LLViewerMedia::onAuthSubmit, _1, _2); +			LLNotifications::instance().add(auth_request_params); +		}; +		break; + +		case MEDIA_EVENT_LINK_HOVERED: +		{ +			LL_DEBUGS("Media") <<  "Media event:  MEDIA_EVENT_LINK_HOVERED, hover text is: " << self->getHoverText() << LL_ENDL; +			mHoverTextChanged = true; +		}; +		break;  	};  	// chain all events to any potential observers of this object. @@ -1098,109 +1083,85 @@ void LLMediaCtrl::onPopup(const LLSD& notification, const LLSD& response)  {  	if (response["open"])  	{ -		LLWeb::loadURL(notification["payload"]["url"], notification["payload"]["target"], notification["payload"]["uuid"]); +		// name of default floater to open +		std::string floater_name = "media_browser"; + +		// look for parent floater name +		if ( gFloaterView ) +		{ +			if ( gFloaterView->getParentFloater(this) ) +			{ +				floater_name = gFloaterView->getParentFloater(this)->getInstanceName(); +			} +			else +			{ +				lldebugs << "No gFloaterView->getParentFloater(this) for onPopuup()" << llendl; +			}; +		} +		else +		{ +			lldebugs << "No gFloaterView for onPopuup()" << llendl; +		}; + +		// (for now) open web content floater if that's our parent, otherwise, open the current media floater +		// (this will change soon) +		if ( floater_name == "web_content" ) +		{ +			LLWeb::loadWebURL(notification["payload"]["url"], notification["payload"]["target"], notification["payload"]["uuid"]); +		} +		else +		{ +			LLWeb::loadURL(notification["payload"]["url"], notification["payload"]["target"], notification["payload"]["uuid"]); +		}  	}  	else  	{  		// Make sure the opening instance knows its window open request was denied, so it can clean things up.  		LLViewerMedia::proxyWindowClosed(notification["payload"]["uuid"]);  	} -  } -void LLMediaCtrl::onCloseNotification() -{ -	LLNotifications::instance().cancel(mCurNotification); -} - -void LLMediaCtrl::onClickIgnore(LLUICtrl* ctrl) +void LLMediaCtrl::showNotification(LLNotificationPtr notify)  { -	bool check = ctrl->getValue().asBoolean(); -	if (mCurNotification && mCurNotification->getForm()->getIgnoreType() == LLNotificationForm::IGNORE_SHOW_AGAIN) +	delete mWindowShade; + +	LLWindowShade::Params params; +	params.name = "notification_shade"; +	params.rect = getLocalRect(); +	params.follows.flags = FOLLOWS_ALL; +	params.notification = notify; +	params.modal = true; +	//HACK: don't hardcode this +	if (notify->getIcon() == "Popup_Caution")  	{ -		// question was "show again" so invert value to get "ignore" -		check = !check; +		params.bg_image.name = "Yellow_Gradient"; +		params.text_color = LLColor4::black;  	} -	mCurNotification->setIgnored(check); -} - -void LLMediaCtrl::onClickNotificationButton(const std::string& name) -{ -	if (!mCurNotification) return; - -	LLSD response = mCurNotification->getResponseTemplate(); -	response[name] = true; - -	mCurNotification->respond(response);  -} - -void LLMediaCtrl::showNotification(LLNotificationPtr notify) -{ -	mCurNotification = notify; - -	// add popup here -	LLSD payload = notify->getPayload(); - -	LLNotificationFormPtr formp = notify->getForm(); -	LLLayoutPanel& panel = getChildRef<LLLayoutPanel>("notification_area"); -	panel.setVisible(true); -	panel.getChild<LLUICtrl>("notification_icon")->setValue(notify->getIcon()); -	panel.getChild<LLUICtrl>("notification_text")->setValue(notify->getMessage()); -	panel.getChild<LLUICtrl>("notification_text")->setToolTip(notify->getMessage()); -	LLNotificationForm::EIgnoreType ignore_type = formp->getIgnoreType();  -	LLLayoutPanel& form_elements = panel.getChildRef<LLLayoutPanel>("form_elements"); -	form_elements.deleteAllChildren(); - -	const S32 FORM_PADDING_HORIZONTAL = 10; -	const S32 FORM_PADDING_VERTICAL = 3; -	S32 cur_x = FORM_PADDING_HORIZONTAL; - -	if (ignore_type != LLNotificationForm::IGNORE_NO) +	else +	//HACK: another one since XUI doesn't support what we need right now +	if (notify->getName() == "AuthRequest")  	{ -		LLCheckBoxCtrl::Params checkbox_p; -		checkbox_p.name = "ignore_check"; -		checkbox_p.rect = LLRect(cur_x, form_elements.getRect().getHeight() - FORM_PADDING_VERTICAL, cur_x, FORM_PADDING_VERTICAL); -		checkbox_p.label = formp->getIgnoreMessage(); -		checkbox_p.label_text.text_color = LLColor4::black; -		checkbox_p.commit_callback.function = boost::bind(&LLMediaCtrl::onClickIgnore, this, _1); -		checkbox_p.initial_value = formp->getIgnored(); - -		LLCheckBoxCtrl* check = LLUICtrlFactory::create<LLCheckBoxCtrl>(checkbox_p); -		check->setRect(check->getBoundingRect()); -		form_elements.addChild(check); -		cur_x = check->getRect().mRight + FORM_PADDING_HORIZONTAL; +		params.bg_image.name = "Yellow_Gradient"; +		params.text_color = LLColor4::black; +		params.can_close = false;  	} - -	for (S32 i = 0; i < formp->getNumElements(); i++) +	else  	{ -		LLSD form_element = formp->getElement(i); -		if (form_element["type"].asString() == "button") -		{ -			LLButton::Params button_p; -			button_p.name = form_element["name"]; -			button_p.label = form_element["text"]; -			button_p.rect = LLRect(cur_x, form_elements.getRect().getHeight() - FORM_PADDING_VERTICAL, cur_x, FORM_PADDING_VERTICAL); -			button_p.click_callback.function = boost::bind(&LLMediaCtrl::onClickNotificationButton, this, form_element["name"].asString()); -			button_p.auto_resize = true; - -			LLButton* button = LLUICtrlFactory::create<LLButton>(button_p); -			button->autoResize(); -			form_elements.addChild(button); - -			cur_x = button->getRect().mRight + FORM_PADDING_HORIZONTAL; -		} +		//HACK: make this a property of the notification itself, "cancellable" +		params.can_close = false; +		params.text_color.control = "LabelTextColor";  	} +	mWindowShade = LLUICtrlFactory::create<LLWindowShade>(params); -	form_elements.reshape(cur_x, form_elements.getRect().getHeight()); - -	//LLWeb::loadURL(payload["url"], payload["target"]); +	addChild(mWindowShade); +	mWindowShade->show();  }  void LLMediaCtrl::hideNotification()  { -	LLLayoutPanel& panel = getChildRef<LLLayoutPanel>("notification_area"); -	panel.setVisible(FALSE); - -	mCurNotification.reset(); +	if (mWindowShade) +	{ +		mWindowShade->hide(); +	}  } diff --git a/indra/newview/llmediactrl.h b/indra/newview/llmediactrl.h index 96bb0c1df5..38a74f90d3 100644 --- a/indra/newview/llmediactrl.h +++ b/indra/newview/llmediactrl.h @@ -90,6 +90,7 @@ public:  		virtual BOOL handleRightMouseUp(S32 x, S32 y, MASK mask);  		virtual BOOL handleDoubleClick( S32 x, S32 y, MASK mask );  		virtual BOOL handleScrollWheel( S32 x, S32 y, S32 clicks ); +		virtual BOOL handleToolTip(S32 x, S32 y, MASK mask);  		// navigation  		void navigateTo( std::string url_in, std::string mime_type = ""); @@ -168,9 +169,6 @@ public:  	private:  		void onVisibilityChange ( const LLSD& new_visibility );  		void onPopup(const LLSD& notification, const LLSD& response); -		void onCloseNotification(); -		void onClickNotificationButton(const std::string& name); -		void onClickIgnore(LLUICtrl* ctrl);  		const S32 mTextureDepthBytes;  		LLUUID mMediaTextureID; @@ -194,7 +192,8 @@ public:  		S32 mTextureWidth;  		S32 mTextureHeight;  		bool mClearCache; -		boost::shared_ptr<class LLNotification> mCurNotification; +		class LLWindowShade* mWindowShade; +		bool mHoverTextChanged;  };  #endif // LL_LLMediaCtrl_H diff --git a/indra/newview/llpanellogin.cpp b/indra/newview/llpanellogin.cpp index 2e4be78be1..457e2d2980 100644 --- a/indra/newview/llpanellogin.cpp +++ b/indra/newview/llpanellogin.cpp @@ -212,9 +212,6 @@ LLPanelLogin::LLPanelLogin(const LLRect &rect,  	//sendChildToBack(getChildView("channel_text"));  	sendChildToBack(getChildView("forgot_password_text")); -	LLLineEditor* edit = getChild<LLLineEditor>("password_edit"); -	if (edit) edit->setDrawAsterixes(TRUE); -  	if(LLStartUp::getStartSLURL().getType() != LLSLURL::LOCATION)  	{  		LLSLURL slurl(gSavedSettings.getString("LoginLocation")); diff --git a/indra/newview/llpanelprimmediacontrols.cpp b/indra/newview/llpanelprimmediacontrols.cpp index 614700fb0a..8ae3553857 100644 --- a/indra/newview/llpanelprimmediacontrols.cpp +++ b/indra/newview/llpanelprimmediacontrols.cpp @@ -59,6 +59,7 @@  #include "llvovolume.h"  #include "llweb.h"  #include "llwindow.h" +#include "llwindowshade.h"  #include "llfloatertools.h"  // to enable hide if build tools are up  // Functions pulled from pipeline.cpp @@ -90,7 +91,8 @@ LLPanelPrimMediaControls::LLPanelPrimMediaControls() :  	mTargetObjectNormal(LLVector3::zero),  	mZoomObjectID(LLUUID::null),  	mZoomObjectFace(0), -	mVolumeSliderVisible(0) +	mVolumeSliderVisible(0), +	mWindowShade(NULL)  {  	mCommitCallbackRegistrar.add("MediaCtrl.Close",		boost::bind(&LLPanelPrimMediaControls::onClickClose, this));  	mCommitCallbackRegistrar.add("MediaCtrl.Back",		boost::bind(&LLPanelPrimMediaControls::onClickBack, this)); @@ -205,6 +207,9 @@ BOOL LLPanelPrimMediaControls::postBuild()  	mMediaAddress->setFocusReceivedCallback(boost::bind(&LLPanelPrimMediaControls::onInputURL, _1, this )); +	LLWindowShade::Params window_shade_params; +	window_shade_params.name = "window_shade"; +  	mCurrentZoom = ZOOM_NONE;  	// clicks on buttons do not remove keyboard focus from media  	setIsChrome(TRUE); @@ -698,6 +703,24 @@ void LLPanelPrimMediaControls::updateShape()  /*virtual*/  void LLPanelPrimMediaControls::draw()  { +	LLViewerMediaImpl* impl = getTargetMediaImpl(); +	if (impl) +	{ +		LLNotificationPtr notification = impl->getCurrentNotification(); +		if (notification != mActiveNotification) +		{ +			mActiveNotification = notification; +			if (notification) +			{ +				showNotification(notification); +			} +			else +			{ +				hideNotification(); +			} +		} +	} +  	F32 alpha = getDrawContext().mAlpha;  	if(mFadeTimer.getStarted())  	{ @@ -1295,3 +1318,38 @@ bool LLPanelPrimMediaControls::shouldVolumeSliderBeVisible()  {  	return mVolumeSliderVisible > 0;  } + +void LLPanelPrimMediaControls::showNotification(LLNotificationPtr notify) +{ +	delete mWindowShade; +	LLWindowShade::Params params; +	params.rect = mMediaRegion->getLocalRect(); +	params.follows.flags = FOLLOWS_ALL; +	params.notification = notify; + +	//HACK: don't hardcode this +	if (notify->getIcon() == "Popup_Caution") +	{ +		params.bg_image.name = "Yellow_Gradient"; +		params.text_color = LLColor4::black; +	} +	else +	{ +		//HACK: make this a property of the notification itself, "cancellable" +		params.can_close = false; +		params.text_color.control = "LabelTextColor"; +	} + +	mWindowShade = LLUICtrlFactory::create<LLWindowShade>(params); + +	mMediaRegion->addChild(mWindowShade); +	mWindowShade->show(); +} + +void LLPanelPrimMediaControls::hideNotification() +{ +	if (mWindowShade) +	{ +		mWindowShade->hide(); +	} +} diff --git a/indra/newview/llpanelprimmediacontrols.h b/indra/newview/llpanelprimmediacontrols.h index 3ec24f0e24..0b9664359c 100644 --- a/indra/newview/llpanelprimmediacontrols.h +++ b/indra/newview/llpanelprimmediacontrols.h @@ -29,6 +29,7 @@  #include "llpanel.h"  #include "llviewermedia.h" +#include "llnotificationptr.h"  class LLButton;  class LLCoordWindow; @@ -37,6 +38,7 @@ class LLLayoutStack;  class LLProgressBar;  class LLSliderCtrl;  class LLViewerMediaImpl; +class LLWindowShade;  class LLPanelPrimMediaControls : public LLPanel  { @@ -54,6 +56,9 @@ public:  	void updateShape();  	bool isMouseOver(); +	void showNotification(LLNotificationPtr notify); +	void hideNotification(); +  	enum EZoomLevel  	{  		ZOOM_NONE = 0, @@ -162,6 +167,7 @@ private:  	LLUICtrl *mRightBookend;  	LLUIImage* mBackgroundImage;  	LLUIImage* mVolumeSliderBackgroundImage; +	LLWindowShade* mWindowShade;  	F32 mSkipStep;  	S32 mMinWidth;  	S32 mMinHeight; @@ -204,6 +210,8 @@ private:  	S32 mZoomObjectFace;  	S32 mVolumeSliderVisible; + +	LLNotificationPtr mActiveNotification;  };  #endif // LL_PANELPRIMMEDIACONTROLS_H diff --git a/indra/newview/llviewerfloaterreg.cpp b/indra/newview/llviewerfloaterreg.cpp index f573f25efe..dca1e33e60 100644 --- a/indra/newview/llviewerfloaterreg.cpp +++ b/indra/newview/llviewerfloaterreg.cpp @@ -60,6 +60,7 @@  #include "llfloaterhardwaresettings.h"  #include "llfloaterhelpbrowser.h"  #include "llfloatermediabrowser.h" +#include "llfloaterwebcontent.h"  #include "llfloatermediasettings.h"  #include "llfloaterhud.h"  #include "llfloaterimagepreview.h" @@ -252,6 +253,7 @@ void LLViewerFloaterReg::registerFloaters()  	LLFloaterReg::add("voice_controls", "floater_voice_controls.xml", (LLFloaterBuildFunc)&LLFloaterReg::build<LLCallFloater>);  	LLFloaterReg::add("voice_effect", "floater_voice_effect.xml", (LLFloaterBuildFunc)&LLFloaterReg::build<LLFloaterVoiceEffect>); +	LLFloaterReg::add("web_content", "floater_web_content.xml", (LLFloaterBuildFunc)&LLFloaterReg::build<LLFloaterWebContent>);	  	LLFloaterReg::add("whitelist_entry", "floater_whitelist_entry.xml", (LLFloaterBuildFunc)&LLFloaterReg::build<LLFloaterWhiteListEntry>);	  	LLFloaterWindowSizeUtil::registerFloater();  	LLFloaterReg::add("world_map", "floater_world_map.xml", (LLFloaterBuildFunc)&LLFloaterReg::build<LLFloaterWorldMap>);	 diff --git a/indra/newview/llviewermedia.cpp b/indra/newview/llviewermedia.cpp index fae4eb3c05..d3b6dcd86f 100644 --- a/indra/newview/llviewermedia.cpp +++ b/indra/newview/llviewermedia.cpp @@ -52,6 +52,7 @@  #include "llviewerregion.h"  #include "llwebsharing.h"	// For LLWebSharing::setOpenIDCookie(), *TODO: find a better way to do this!  #include "llfilepicker.h" +#include "llnotifications.h"  #include "llevent.h"		// LLSimpleListener  #include "llnotificationsutil.h" @@ -62,6 +63,7 @@  #include "llwindow.h"  #include "llfloatermediabrowser.h"	// for handling window close requests and geometry change requests in media browser windows. +#include "llfloaterwebcontent.h"	// for handling window close requests and geometry change requests in media browser windows.  #include <boost/bind.hpp>	// for SkinFolder listener  #include <boost/signals2.hpp> @@ -293,6 +295,7 @@ public:  LLPluginCookieStore *LLViewerMedia::sCookieStore = NULL;  LLURL LLViewerMedia::sOpenIDURL;  std::string LLViewerMedia::sOpenIDCookie; +LLPluginClassMedia* LLViewerMedia::sSpareBrowserMediaSource = NULL;  static LLViewerMedia::impl_list sViewerMediaImplList;  static LLViewerMedia::impl_id_map sViewerMediaTextureIDMap;  static LLTimer sMediaCreateTimer; @@ -742,6 +745,9 @@ void LLViewerMedia::updateMedia(void *dummy_arg)  	// Enable/disable the plugin read thread  	LLPluginProcessParent::setUseReadThread(gSavedSettings.getBOOL("PluginUseReadThread")); +	// HACK: we always try to keep a spare running webkit plugin around to improve launch times. +	createSpareBrowserMediaSource(); +	  	sAnyMediaShowing = false;  	sUpdatedCookies = getCookieStore()->getChangedCookies();  	if(!sUpdatedCookies.empty()) @@ -759,6 +765,12 @@ void LLViewerMedia::updateMedia(void *dummy_arg)  		pimpl->update();  		pimpl->calculateInterest();  	} +	 +	// Let the spare media source actually launch +	if(sSpareBrowserMediaSource) +	{ +		sSpareBrowserMediaSource->idle(); +	}  	// Sort the static instance list using our interest criteria  	sViewerMediaImplList.sort(priorityComparitor); @@ -1034,6 +1046,26 @@ bool LLViewerMedia::isParcelAudioPlaying()  	return (LLViewerMedia::hasParcelAudio() && gAudiop && LLAudioEngine::AUDIO_PLAYING == gAudiop->isInternetStreamPlaying());  } +void LLViewerMedia::onAuthSubmit(const LLSD& notification, const LLSD& response) +{ +	LLViewerMediaImpl *impl = LLViewerMedia::getMediaImplFromTextureID(notification["payload"]["media_id"]); +	if(impl) +	{ +		LLPluginClassMedia* media = impl->getMediaPlugin(); +		if(media) +		{ +			if (response["ok"]) +			{ +				media->sendAuthResponse(true, response["username"], response["password"]); +			} +			else +			{ +				media->sendAuthResponse(false, "", ""); +			} +		} +	} +} +  /////////////////////////////////////////////////////////////////////////////////////////  // static  void LLViewerMedia::clearAllCookies() @@ -1400,6 +1432,29 @@ void LLViewerMedia::proxyWindowClosed(const std::string &uuid)  	}  } +///////////////////////////////////////////////////////////////////////////////////////// +// static +void LLViewerMedia::createSpareBrowserMediaSource() +{ +	if(!sSpareBrowserMediaSource) +	{ +		// If we don't have a spare browser media source, create one. +		// The null owner will keep the browser plugin from fully initializing  +		// (specifically, it keeps LLPluginClassMedia from negotiating a size change,  +		// which keeps MediaPluginWebkit::initBrowserWindow from doing anything until we have some necessary data, like the background color) +		sSpareBrowserMediaSource = LLViewerMediaImpl::newSourceFromMediaType("text/html", NULL, 0, 0); +	} +} + +///////////////////////////////////////////////////////////////////////////////////////// +// static +LLPluginClassMedia* LLViewerMedia::getSpareBrowserMediaSource()  +{ +	LLPluginClassMedia* result = sSpareBrowserMediaSource; +	sSpareBrowserMediaSource = NULL; +	return result;  +}; +  bool LLViewerMedia::hasInWorldMedia()  {  	if (sInWorldMediaDisabled) return false; @@ -1636,6 +1691,21 @@ void LLViewerMediaImpl::setMediaType(const std::string& media_type)  LLPluginClassMedia* LLViewerMediaImpl::newSourceFromMediaType(std::string media_type, LLPluginClassMediaOwner *owner /* may be NULL */, S32 default_width, S32 default_height, const std::string target)  {  	std::string plugin_basename = LLMIMETypes::implType(media_type); +	LLPluginClassMedia* media_source = NULL; +	 +	// HACK: we always try to keep a spare running webkit plugin around to improve launch times. +	if(plugin_basename == "media_plugin_webkit") +	{ +		media_source = LLViewerMedia::getSpareBrowserMediaSource(); +		if(media_source) +		{ +			media_source->setOwner(owner); +			media_source->setTarget(target); +			media_source->setSize(default_width, default_height); +						 +			return media_source; +		} +	}  	if(plugin_basename.empty())  	{ @@ -1673,7 +1743,7 @@ LLPluginClassMedia* LLViewerMediaImpl::newSourceFromMediaType(std::string media_  		}  		else  		{ -			LLPluginClassMedia* media_source = new LLPluginClassMedia(owner); +			media_source = new LLPluginClassMedia(owner);  			media_source->setSize(default_width, default_height);  			media_source->setUserDataPath(user_data_path);  			media_source->setLanguageCode(LLUI::getLanguage()); @@ -1753,6 +1823,22 @@ bool LLViewerMediaImpl::initializePlugin(const std::string& media_type)  		media_source->focus(mHasFocus);  		media_source->setBackgroundColor(mBackgroundColor); +		if(gSavedSettings.getBOOL("BrowserIgnoreSSLCertErrors")) +		{ +			media_source->ignore_ssl_cert_errors(true); +		} + +		// start by assuming the default CA file will be used +		std::string ca_path = gDirUtilp->getExpandedFilename( LL_PATH_APP_SETTINGS, "lindenlab.pem" ); +	 +		// default turned off so pick up the user specified path +		if( ! gSavedSettings.getBOOL("BrowserUseDefaultCAFile")) +		{ +			ca_path = gSavedSettings.getString("BrowserCAFilePath"); +		} +		// set the path to the CA.pem file +		media_source->addCertificateFilePath( ca_path ); +  		media_source->proxy_setup(gSavedSettings.getBOOL("BrowserProxyEnabled"), gSavedSettings.getString("BrowserProxyAddress"), gSavedSettings.getS32("BrowserProxyPort"));  		if(mClearCache) @@ -1849,6 +1935,18 @@ void LLViewerMediaImpl::setSize(int width, int height)  }  ////////////////////////////////////////////////////////////////////////////////////////// +void LLViewerMediaImpl::showNotification(LLNotificationPtr notify) +{ +	mNotification = notify; +} + +////////////////////////////////////////////////////////////////////////////////////////// +void LLViewerMediaImpl::hideNotification() +{ +	mNotification.reset(); +} + +//////////////////////////////////////////////////////////////////////////////////////////  void LLViewerMediaImpl::play()  {  	// If the media source isn't there, try to initialize it and load an URL. @@ -2850,7 +2948,6 @@ void LLViewerMediaImpl::handleMediaEvent(LLPluginClassMedia* plugin, LLPluginCla  			LL_DEBUGS("Media") << "MEDIA_EVENT_CLICK_LINK_NOFOLLOW, uri is: " << plugin->getClickURL() << LL_ENDL;   			std::string url = plugin->getClickURL();  			LLURLDispatcher::dispatch(url, NULL, mTrustedBrowser); -  		}  		break;  		case MEDIA_EVENT_CLICK_LINK_HREF: @@ -2913,6 +3010,7 @@ void LLViewerMediaImpl::handleMediaEvent(LLPluginClassMedia* plugin, LLPluginCla  		case LLViewerMediaObserver::MEDIA_EVENT_NAVIGATE_BEGIN:  		{  			LL_DEBUGS("Media") << "MEDIA_EVENT_NAVIGATE_BEGIN, uri is: " << plugin->getNavigateURI() << LL_ENDL; +			hideNotification();  			if(getNavState() == MEDIANAVSTATE_SERVER_SENT)  			{ @@ -3003,7 +3101,26 @@ void LLViewerMediaImpl::handleMediaEvent(LLPluginClassMedia* plugin, LLPluginCla  			plugin->sendPickFileResponse(response);  		}  		break; -		 + + +		case LLViewerMediaObserver::MEDIA_EVENT_AUTH_REQUEST: +		{ +			LLNotification::Params auth_request_params; +			auth_request_params.name = "AuthRequest"; + +			// pass in host name and realm for site (may be zero length but will always exist) +			LLSD args; +			LLURL raw_url( plugin->getAuthURL().c_str() ); +			args["HOST_NAME"] = raw_url.getAuthority(); +			args["REALM"] = plugin->getAuthRealm(); +			auth_request_params.substitutions = args; + +			auth_request_params.payload = LLSD().with("media_id", mTextureId); +			auth_request_params.functor.function = boost::bind(&LLViewerMedia::onAuthSubmit, _1, _2); +			LLNotifications::instance().add(auth_request_params); +		}; +		break; +  		case LLViewerMediaObserver::MEDIA_EVENT_CLOSE_REQUEST:  		{  			std::string uuid = plugin->getClickUUID(); @@ -3019,6 +3136,7 @@ void LLViewerMediaImpl::handleMediaEvent(LLPluginClassMedia* plugin, LLPluginCla  				// This close request is directed at another instance  				pass_through = false;  				LLFloaterMediaBrowser::closeRequest(uuid); +				LLFloaterWebContent::closeRequest(uuid);  			}  		}  		break; @@ -3038,6 +3156,7 @@ void LLViewerMediaImpl::handleMediaEvent(LLPluginClassMedia* plugin, LLPluginCla  				// This request is directed at another instance  				pass_through = false;  				LLFloaterMediaBrowser::geometryChanged(uuid, plugin->getGeometryX(), plugin->getGeometryY(), plugin->getGeometryWidth(), plugin->getGeometryHeight()); +				LLFloaterWebContent::geometryChanged(uuid, plugin->getGeometryX(), plugin->getGeometryY(), plugin->getGeometryWidth(), plugin->getGeometryHeight());  			}  		}  		break; @@ -3521,6 +3640,11 @@ bool LLViewerMediaImpl::isInAgentParcel() const  	return result;  } +LLNotificationPtr LLViewerMediaImpl::getCurrentNotification() const +{ +	return mNotification; +} +  //////////////////////////////////////////////////////////////////////////////////////////  //  // static diff --git a/indra/newview/llviewermedia.h b/indra/newview/llviewermedia.h index 4025a4484f..e2e342cc45 100644 --- a/indra/newview/llviewermedia.h +++ b/indra/newview/llviewermedia.h @@ -37,6 +37,7 @@  #include "llpluginclassmedia.h"  #include "v4color.h" +#include "llnotificationptr.h"  #include "llurl.h" @@ -130,6 +131,8 @@ public:  	static bool isParcelMediaPlaying();  	static bool isParcelAudioPlaying(); +	static void onAuthSubmit(const LLSD& notification, const LLSD& response); +  	// Clear all cookies for all plugins  	static void clearAllCookies(); @@ -155,6 +158,9 @@ public:  	static void proxyWindowOpened(const std::string &target, const std::string &uuid);  	static void proxyWindowClosed(const std::string &uuid); +	static void createSpareBrowserMediaSource(); +	static LLPluginClassMedia* getSpareBrowserMediaSource(); +	  private:  	static void setOpenIDCookie();  	static void onTeleportFinished(); @@ -162,6 +168,7 @@ private:  	static LLPluginCookieStore *sCookieStore;  	static LLURL sOpenIDURL;  	static std::string sOpenIDCookie; +	static LLPluginClassMedia* sSpareBrowserMediaSource;  };  // Implementation functions not exported into header file @@ -195,6 +202,9 @@ public:  	LLPluginClassMedia* getMediaPlugin() { return mMediaSource; }  	void setSize(int width, int height); +	void showNotification(LLNotificationPtr notify); +	void hideNotification(); +  	void play();  	void stop();  	void pause(); @@ -387,6 +397,9 @@ public:  	// Is this media in the agent's parcel?  	bool isInAgentParcel() const; +	// get currently active notification associated with this media instance +	LLNotificationPtr getCurrentNotification() const; +  private:  	bool isAutoPlayable() const;  	bool shouldShowBasedOnClass() const; @@ -444,7 +457,8 @@ private:  	bool mNavigateSuspendedDeferred;  	bool mTrustedBrowser;  	std::string mTarget; -	 +	LLNotificationPtr mNotification; +  private:  	BOOL mIsUpdated ;  	std::list< LLVOVolume* > mObjectList ; diff --git a/indra/newview/llviewermenu.cpp b/indra/newview/llviewermenu.cpp index e18b4ec414..9e16bf2fbb 100644 --- a/indra/newview/llviewermenu.cpp +++ b/indra/newview/llviewermenu.cpp @@ -7223,6 +7223,12 @@ void handle_web_browser_test(const LLSD& param)  	LLWeb::loadURLInternal(url);  } +void handle_web_content_test(const LLSD& param) +{ +	std::string url = param.asString(); +	LLWeb::loadWebURLInternal(url); +} +  void handle_buy_currency_test(void*)  {  	std::string url = @@ -7973,7 +7979,8 @@ void initialize_menus()  	view_listener_t::addMenu(new LLAdvancedDumpRegionObjectCache(), "Advanced.DumpRegionObjectCache");  	// Advanced > UI -	commit.add("Advanced.WebBrowserTest", boost::bind(&handle_web_browser_test, _2)); +	commit.add("Advanced.WebBrowserTest", boost::bind(&handle_web_browser_test,	_2));	// sigh! this one opens the MEDIA browser +	commit.add("Advanced.WebContentTest", boost::bind(&handle_web_content_test, _2));	// this one opens the Web Content floater  	view_listener_t::addMenu(new LLAdvancedBuyCurrencyTest(), "Advanced.BuyCurrencyTest");  	view_listener_t::addMenu(new LLAdvancedDumpSelectMgr(), "Advanced.DumpSelectMgr");  	view_listener_t::addMenu(new LLAdvancedDumpInventory(), "Advanced.DumpInventory"); diff --git a/indra/newview/llviewerparcelmedia.cpp b/indra/newview/llviewerparcelmedia.cpp index 99e869dafc..40f0b43313 100644 --- a/indra/newview/llviewerparcelmedia.cpp +++ b/indra/newview/llviewerparcelmedia.cpp @@ -586,6 +586,18 @@ void LLViewerParcelMedia::handleMediaEvent(LLPluginClassMedia* self, EMediaEvent  			LL_DEBUGS("Media") << "Media event:  MEDIA_EVENT_GEOMETRY_CHANGE, uuid is " << self->getClickUUID() << LL_ENDL;  		}  		break; + +		case MEDIA_EVENT_AUTH_REQUEST: +		{ +			LL_DEBUGS("Media") <<  "Media event:  MEDIA_EVENT_AUTH_REQUEST, url " << self->getAuthURL() << ", realm " << self->getAuthRealm() << LL_ENDL; +		} +		break; + +		case MEDIA_EVENT_LINK_HOVERED: +		{ +			LL_DEBUGS("Media") <<  "Media event:  MEDIA_EVENT_LINK_HOVERED, hover text is: " << self->getHoverText() << LL_ENDL; +		}; +		break;  	};  } diff --git a/indra/newview/llweb.cpp b/indra/newview/llweb.cpp index 6028a8fbea..b73017a51a 100644 --- a/indra/newview/llweb.cpp +++ b/indra/newview/llweb.cpp @@ -35,6 +35,7 @@  #include "llagent.h"  #include "llappviewer.h"  #include "llfloatermediabrowser.h" +#include "llfloaterwebcontent.h"  #include "llfloaterreg.h"  #include "lllogininstance.h"  #include "llparcel.h" @@ -95,6 +96,23 @@ void LLWeb::loadURL(const std::string& url, const std::string& target, const std  	}  } +// static +void LLWeb::loadWebURL(const std::string& url, const std::string& target, const std::string& uuid) +{ +	if(target == "_internal") +	{ +		// Force load in the internal browser, as if with a blank target. +		loadWebURLInternal(url, "", uuid); +	} +	else if (gSavedSettings.getBOOL("UseExternalBrowser") || (target == "_external")) +	{ +		loadURLExternal(url); +	} +	else +	{ +		loadWebURLInternal(url, target, uuid); +	} +}  // static  void LLWeb::loadURLInternal(const std::string &url, const std::string& target, const std::string& uuid) @@ -102,6 +120,13 @@ void LLWeb::loadURLInternal(const std::string &url, const std::string& target, c  	LLFloaterMediaBrowser::create(url, target, uuid);  } +// static +// Explicitly open a Web URL using the Web content floater +void LLWeb::loadWebURLInternal(const std::string &url, const std::string& target, const std::string& uuid) +{ +	LLFloaterWebContent::create(url, target, uuid); +} +  // static  void LLWeb::loadURLExternal(const std::string& url, const std::string& uuid) diff --git a/indra/newview/llweb.h b/indra/newview/llweb.h index 2915376583..dc5958e57f 100644 --- a/indra/newview/llweb.h +++ b/indra/newview/llweb.h @@ -57,6 +57,11 @@ public:  	static void loadURLExternal(const std::string& url, const std::string& uuid);  	static void loadURLExternal(const std::string& url, bool async, const std::string& uuid = LLStringUtil::null); +	// Explicitly open a Web URL using the Web content floater vs. the more general media browser +	static void loadWebURL(const std::string& url, const std::string& target, const std::string& uuid); +	static void loadWebURLInternal(const std::string &url, const std::string& target, const std::string& uuid); +	static void loadWebURLInternal(const std::string &url) { loadWebURLInternal(url, LLStringUtil::null, LLStringUtil::null); } +  	/// Returns escaped url (eg, " " to "%20") - used by all loadURL methods  	static std::string escapeURL(const std::string& url);  	/// Expands various strings like [LANG], [VERSION], etc. in a URL diff --git a/indra/newview/skins/default/textures/textures.xml b/indra/newview/skins/default/textures/textures.xml index a51a096482..89611d8899 100644 --- a/indra/newview/skins/default/textures/textures.xml +++ b/indra/newview/skins/default/textures/textures.xml @@ -392,7 +392,7 @@ with the same filename but different name    <texture name="RadioButton_On_Disabled" file_name="widgets/RadioButton_On_Disabled.png" preload="true" /> -  <texture name="Refresh_Off" file_name="icons/Refresh_Off.png" preload="false" /> +  <texture name="Refresh_Off" file_name="icons/Refresh_Off.png" preload="true" />    <texture name="Resize_Corner" file_name="windows/Resize_Corner.png" preload="true" /> @@ -468,7 +468,7 @@ with the same filename but different name    <texture name="Stepper_Up_Off" file_name="widgets/Stepper_Up_Off.png" preload="false" />    <texture name="Stepper_Up_Press" file_name="widgets/Stepper_Up_Press.png" preload="false" /> -  <texture name="Stop_Off" file_name="icons/Stop_Off.png" preload="false" /> +  <texture name="Stop_Off" file_name="icons/Stop_Off.png" preload="true" />    <texture name="StopReload_Off" file_name="icons/StopReload_Off.png" preload="false" />    <texture name="StopReload_Over" file_name="icons/StopReload_Over.png" preload="false" /> diff --git a/indra/newview/skins/default/xui/en/floater_help_browser.xml b/indra/newview/skins/default/xui/en/floater_help_browser.xml index 837923bcf6..02e50ee584 100644 --- a/indra/newview/skins/default/xui/en/floater_help_browser.xml +++ b/indra/newview/skins/default/xui/en/floater_help_browser.xml @@ -36,7 +36,8 @@           user_resize="false"           width="620">              <web_browser -              trusted_content="true"  +             trusted_content="true"  +             initial_mime_type="text/html"                bottom="-25"               follows="left|right|top|bottom"               layout="topleft" diff --git a/indra/newview/skins/default/xui/en/floater_web_content.xml b/indra/newview/skins/default/xui/en/floater_web_content.xml new file mode 100644 index 0000000000..2ad46824c2 --- /dev/null +++ b/indra/newview/skins/default/xui/en/floater_web_content.xml @@ -0,0 +1,190 @@ +<?xml version="1.0" encoding="utf-8" standalone="yes" ?>
 +<floater
 +  legacy_header_height="18"
 +  can_resize="true"
 +  height="440"
 +  layout="topleft"
 +  min_height="140"
 +  min_width="467"
 +  name="floater_web_content"
 +  help_topic="floater_web_content"
 +  save_rect="true"
 +  auto_tile="true"
 +  title=""
 +  initial_mime_type="text/html"
 +  width="820">
 +  <layout_stack
 +    bottom="440"
 +    follows="left|right|top|bottom"
 +    layout="topleft"
 +    left="5"
 +    name="stack1"
 +    orientation="vertical"
 +    top="20"
 +    width="810">
 +    <layout_panel
 +      auto_resize="false"
 +      default_tab_group="1"
 +      height="22"
 +      layout="topleft"
 +      left="0"
 +      min_height="20"
 +      name="nav_controls"
 +      top="400"
 +      user_resize="false"
 +      width="800">
 +      <button
 +        image_overlay="Arrow_Left_Off"
 +		    image_disabled="PushButton_Disabled"
 +		    image_disabled_selected="PushButton_Disabled"
 +		    image_selected="PushButton_Selected"
 +		    image_unselected="PushButton_Off"
 +		    hover_glow_amount="0.15"
 +        tool_tip="Navigate back"
 +        follows="left|top"
 +        height="22"
 +        layout="topleft"
 +        left="1"
 +        name="back"
 +        top="0"
 +        width="22">
 +        <button.commit_callback
 +          function="WebContent.Back" />
 +      </button>
 +      <button
 +        image_overlay="Arrow_Right_Off"
 +		    image_disabled="PushButton_Disabled"
 +		    image_disabled_selected="PushButton_Disabled"
 +		    image_selected="PushButton_Selected"
 +		    image_unselected="PushButton_Off"
 +        tool_tip="Navigate forward"
 +        follows="left|top"
 +        height="22"
 +        layout="topleft"
 +        left="27"
 +        name="forward"
 +        top_delta="0"
 +        width="22">
 +        <button.commit_callback
 +          function="WebContent.Forward" />
 +      </button>
 +      <button
 +        image_overlay="Stop_Off"
 +		    image_disabled="PushButton_Disabled"
 +		    image_disabled_selected="PushButton_Disabled"
 +		    image_selected="PushButton_Selected"
 +		    image_unselected="PushButton_Off"
 +        tool_tip="Stop navigation"
 +        enabled="true"
 +        follows="left|top"
 +        height="22"
 +        layout="topleft"
 +        left="51"
 +        name="stop"
 +        top_delta="0"
 +        width="22">
 +        <button.commit_callback
 +          function="WebContent.Stop" />
 +      </button>
 +      <button
 +        image_overlay="Refresh_Off"
 +		    image_disabled="PushButton_Disabled"
 +		    image_disabled_selected="PushButton_Disabled"
 +		    image_selected="PushButton_Selected"
 +		    image_unselected="PushButton_Off"
 +        tool_tip="Reload page"
 +        follows="left|top"
 +        height="22"
 +        layout="topleft"
 +        left="51"
 +        name="reload"
 +        top_delta="0"
 +        width="22">
 +        <button.commit_callback
 +          function="WebContent.Reload" />
 +      </button>
 +      <combo_box
 +        allow_text_entry="true"
 +        follows="left|top|right"
 +        tab_group="1"
 +        height="22"
 +        layout="topleft"
 +        left_pad="4"
 +        max_chars="1024"
 +        name="address"
 +        combo_editor.select_on_focus="true"
 +        tool_tip="Enter URL here"
 +        top_delta="0"
 +        width="702">
 +        <combo_box.commit_callback
 +          function="WebContent.EnterAddress" />
 +      </combo_box>
 +      <icon
 +        name="media_secure_lock_flag"
 +        height="16"
 +        follows="top|right"
 +        image_name="Lock2"
 +        layout="topleft"
 +        left_delta="656"
 +        top_delta="2"
 +        visible="false" 
 +        tool_tip="Secured Browsing"
 +        width="16" />
 +      <button
 +        image_overlay="ExternalBrowser_Off"
 +		    image_disabled="PushButton_Disabled"
 +		    image_disabled_selected="PushButton_Disabled"
 +		    image_selected="PushButton_Selected"
 +		    image_unselected="PushButton_Off"
 +        tool_tip="Open current URL in your desktop browser"
 +        follows="right|top"
 +        enabled="true" 
 +        height="22"
 +        layout="topleft"
 +        name="popexternal"
 +        right="800"
 +        top_delta="-2"
 +        width="22">
 +        <button.commit_callback
 +          function="WebContent.PopExternal" />
 +      </button>
 +    </layout_panel>
 +    <layout_panel
 +      height="40"
 +      layout="topleft"
 +      left_delta="0"
 +      name="external_controls"
 +      top_delta="0"
 +      user_resize="false"
 +      width="540">
 +      <web_browser
 +        bottom="-22"
 +        follows="all"
 +        layout="topleft"
 +        left="0"
 +        name="webbrowser"
 +        top="0"/>
 +      <text
 +        type="string"
 +        length="100"
 +        follows="bottom|left"
 +        height="20"
 +        layout="topleft"
 +        left_delta="0"
 +        name="statusbartext"
 +        parse_urls="false"
 +        text_color="0.4 0.4 0.4 1" 
 +        top_pad="5"
 +        width="452"/>
 +      <progress_bar
 +        color_bar="0.3 1.0 0.3 1"
 +        follows="bottom|right"
 +        height="16"
 +        top_delta="-1"
 +        left_pad="24"
 +        layout="topleft"
 +        name="statusbarprogress"
 +        width="64"/>
 +    </layout_panel>
 +  </layout_stack>
 +</floater>
 diff --git a/indra/newview/skins/default/xui/en/menu_login.xml b/indra/newview/skins/default/xui/en/menu_login.xml index 4f982cc8e9..0d4a095e14 100644 --- a/indra/newview/skins/default/xui/en/menu_login.xml +++ b/indra/newview/skins/default/xui/en/menu_login.xml @@ -176,13 +176,19 @@               parameter="message_critical" />          </menu_item_call>          <menu_item_call -         label="Web Browser Test" +         label="Media Browser Test"           name="Web Browser Test">            <menu_item_call.on_click             function="Advanced.WebBrowserTest"             parameter="http://join.secondlife.com/"/>          </menu_item_call> -      <menu_item_separator/> +      <menu_item_call +       label="Web Content Floater Test" +       name="Web Content Floater Test"> +        <menu_item_call.on_click +         function="Advanced.WebContentTest" +         parameter="http://www.google.com"/> +      </menu_item_call>        <menu_item_check          label="Show Grid Picker"          name="Show Grid Picker" diff --git a/indra/newview/skins/default/xui/en/menu_viewer.xml b/indra/newview/skins/default/xui/en/menu_viewer.xml index 3b1ebc64ab..d997a262a8 100644 --- a/indra/newview/skins/default/xui/en/menu_viewer.xml +++ b/indra/newview/skins/default/xui/en/menu_viewer.xml @@ -2646,13 +2646,21 @@                   parameter="BottomPanelNew" />              </menu_item_check>-->              <menu_item_call -             label="Web Browser Test" +             label="Media Browser Test"               name="Web Browser Test">                  <menu_item_call.on_click                   function="Advanced.WebBrowserTest"                   parameter="http://secondlife.com/app/search/slurls.html"/>              </menu_item_call> -            <menu_item_call +          <menu_item_call +           label="Web Content Browser" +           name="Web Content Browser" +           shortcut="control|alt|W"> +            <menu_item_call.on_click +             function="Advanced.WebContentTest" +             parameter="http://google.com"/> +          </menu_item_call> +          <menu_item_call               label="Dump SelectMgr"               name="Dump SelectMgr">                  <menu_item_call.on_click diff --git a/indra/newview/skins/default/xui/en/notifications.xml b/indra/newview/skins/default/xui/en/notifications.xml index b1fd579c6f..11a4970488 100644 --- a/indra/newview/skins/default/xui/en/notifications.xml +++ b/indra/newview/skins/default/xui/en/notifications.xml @@ -5014,7 +5014,7 @@ If you want to view streaming media on parcels that support it you should go to     type="notify">  No Media Plugin was found to handle the "[MIME_TYPE]" mime type.  Media of this type will be unavailable.      <unique> -      <context key="[MIME_TYPE]"/> +      <context>MIME_TYPE</context>      </unique>    </notification> @@ -5943,7 +5943,7 @@ You may only select up to [MAX_SELECT] items from this list.  [NAME] is inviting you to a Voice Chat call.  Click Accept to join the call or Decline to decline the invitation. Click Block to block this caller.      <unique> -      <context key="NAME"/> +      <context>NAME</context>      </unique>      <form name="form">        <button @@ -5992,8 +5992,8 @@ Click Accept to join the call or Decline to decline the invitation. Click Block  [NAME] has joined a Voice Chat call with the group [GROUP].  Click Accept to join the call or Decline to decline the invitation. Click Block to block this caller.      <unique> -      <context key="NAME"/> -      <context key="GROUP"/> +      <context>NAME</context> +      <context>GROUP</context>      </unique>      <form name="form">        <button @@ -6018,7 +6018,7 @@ Click Accept to join the call or Decline to decline the invitation. Click Block  [NAME] has joined a voice chat call with a conference chat.  Click Accept to join the call or Decline to decline the invitation. Click Block to block this caller.      <unique> -      <context key="NAME"/> +      <context>NAME</context>      </unique>      <form name="form">        <button @@ -6043,7 +6043,7 @@ Click Accept to join the call or Decline to decline the invitation. Click Block  [NAME] is inviting you to a conference chat.  Click Accept to join the chat or Decline to decline the invitation. Click Block to block this caller.      <unique> -      <context key="NAME"/> +      <context>NAME</context>      </unique>      <form name="form">        <button @@ -6067,7 +6067,7 @@ Click Accept to join the chat or Decline to decline the invitation. Click Block     type="notifytip">  The voice call you are trying to join, [VOICE_CHANNEL_NAME], has reached maximum capacity. Please try again later.      <unique> -      <context key="VOICE_CHANNEL_NAME"/> +      <context>VOICE_CHANNEL_NAME</context>      </unique>    </notification> @@ -6085,7 +6085,7 @@ We're sorry.  This area has reached maximum capacity for voice conversation     type="notifytip">  You have been disconnected from [VOICE_CHANNEL_NAME].  You will now be reconnected to Nearby Voice Chat.      <unique> -      <context key="VOICE_CHANNEL_NAME"/> +      <context>VOICE_CHANNEL_NAME</context>      </unique>    </notification> @@ -6095,7 +6095,7 @@ You have been disconnected from [VOICE_CHANNEL_NAME].  You will now be reconnect     type="notifytip">  [VOICE_CHANNEL_NAME] has ended the call.  You will now be reconnected to Nearby Voice Chat.      <unique> -      <context key="VOICE_CHANNEL_NAME"/> +      <context>VOICE_CHANNEL_NAME</context>      </unique>    </notification> @@ -6105,7 +6105,7 @@ You have been disconnected from [VOICE_CHANNEL_NAME].  You will now be reconnect     type="notifytip">  [VOICE_CHANNEL_NAME] has declined your call.  You will now be reconnected to Nearby Voice Chat.      <unique> -      <context key="VOICE_CHANNEL_NAME"/> +      <context>VOICE_CHANNEL_NAME</context>      </unique>    </notification> @@ -6115,7 +6115,7 @@ You have been disconnected from [VOICE_CHANNEL_NAME].  You will now be reconnect     type="notifytip">  [VOICE_CHANNEL_NAME] is not available to take your call.  You will now be reconnected to Nearby Voice Chat.      <unique> -      <context key="VOICE_CHANNEL_NAME"/> +      <context>VOICE_CHANNEL_NAME</context>      </unique>    </notification> @@ -6125,7 +6125,7 @@ You have been disconnected from [VOICE_CHANNEL_NAME].  You will now be reconnect     type="notifytip">  Failed to connect to [VOICE_CHANNEL_NAME], please try again later.  You will now be reconnected to Nearby Voice Chat.      <unique> -      <context key="VOICE_CHANNEL_NAME"/> +      <context>VOICE_CHANNEL_NAME</context>      </unique>    </notification> @@ -6211,7 +6211,7 @@ Cannot enter parcel, you are not on the access list.     type="notifytip">  You do not have permission to connect to voice chat for [VOICE_CHANNEL_NAME].      <unique> -      <context key="VOICE_CHANNEL_NAME"/> +      <context>VOICE_CHANNEL_NAME</context>      </unique>    </notification> @@ -6221,7 +6221,7 @@ You do not have permission to connect to voice chat for [VOICE_CHANNEL_NAME].     type="notifytip">  An error has occurred while trying to connect to voice chat for [VOICE_CHANNEL_NAME].  Please try again later.      <unique> -      <context key="VOICE_CHANNEL_NAME"/> +      <context>VOICE_CHANNEL_NAME</context>      </unique>    </notification> @@ -6628,6 +6628,23 @@ Mute everyone?      </form>    </notification> +  <notification +  name="AuthRequest" +  type="browser"> +The site at '<nolink>[HOST_NAME]</nolink>' in realm '[REALM]' requires a user name and password. +    <form name="form"> +      <input name="username" type="text" text="User Name"/> +      <input name="password" type="password" text="Password    "/> +      <button default="true" +              index="0" +              name="ok" +              text="Submit"/> +      <button index="1" +              name="cancel" +              text="Cancel"/> +    </form> +  </notification> +    <global name="UnsupportedCPU">  - Your CPU speed does not meet the minimum requirements. diff --git a/indra/newview/skins/default/xui/en/panel_login.xml b/indra/newview/skins/default/xui/en/panel_login.xml index e3cd61c5aa..b5c584920f 100644 --- a/indra/newview/skins/default/xui/en/panel_login.xml +++ b/indra/newview/skins/default/xui/en/panel_login.xml @@ -92,6 +92,7 @@ follows="left|bottom"    height="22"    max_length_bytes="16"  name="password_edit" +is_password="true"   select_on_focus="true"    top_pad="0"    width="135" /> diff --git a/indra/test_apps/llplugintest/llmediaplugintest.cpp b/indra/test_apps/llplugintest/llmediaplugintest.cpp index 873fa23db8..4a2272032b 100644 --- a/indra/test_apps/llplugintest/llmediaplugintest.cpp +++ b/indra/test_apps/llplugintest/llmediaplugintest.cpp @@ -2220,6 +2220,21 @@ void LLMediaPluginTest::handleMediaEvent(LLPluginClassMedia* self, EMediaEvent e  				<< ", height = " << self->getGeometryHeight()   				<< std::endl;  		break; + +		case MEDIA_EVENT_AUTH_REQUEST: +		{ +			//std::cerr <<  "Media event:  MEDIA_EVENT_AUTH_REQUEST, url " << self->getAuthURL() ", realm " << self->getAuthRealm() << std::endl; + +			// TODO: display an auth dialog +			self->sendAuthResponse(false, "", ""); +		} +		break; + +		case MEDIA_EVENT_LINK_HOVERED: +		{ +			std::cerr <<  "Media event:  MEDIA_EVENT_LINK_HOVERED, hover text is: " << self->getHoverText() << std::endl; +		}; +		break;  	}  } diff --git a/install.xml b/install.xml index 36e7824e1c..13abaac1c1 100644 --- a/install.xml +++ b/install.xml @@ -981,9 +981,9 @@ anguage Infrstructure (CLI) international standard</string>            <key>darwin</key>            <map>              <key>md5sum</key> -            <string>34d9e4c93678a422cf80521bf0cd7628</string> +            <string>66c46841825ab4969ec875b5c8f9b24c</string>              <key>url</key> -            <uri>http://s3.amazonaws.com/viewer-source-downloads/install_pkgs/llqtwebkit-4.6-darwin-20100914.tar.bz2</uri> +            <uri>http://viewer-source-downloads.s3.amazonaws.com/install_pkgs/llqtwebkit-darwin-qt4.7.1-20101221.tar.bz2</uri>            </map>            <key>linux</key>            <map> @@ -995,9 +995,9 @@ anguage Infrstructure (CLI) international standard</string>            <key>windows</key>            <map>              <key>md5sum</key> -            <string>4b8412833c00f8cdaba26808f0ddb404</string> +            <string>b678c4d18ea8e4fab42b20f8d0b2629a</string>              <key>url</key> -            <uri>http://viewer-source-downloads.s3.amazonaws.com/install_pkgs/llqtwebkit-windows-qt4.6-20100916.tar.bz2</uri> +            <uri>http://viewer-source-downloads.s3.amazonaws.com/install_pkgs/llqtwebkit-windows-qt4.7.1-20101221.tar.bz2</uri>            </map>          </map>        </map> | 
