|  | /*
  ==============================================================================
   This file is part of the JUCE library - "Jules' Utility Class Extensions"
   Copyright 2004-7 by Raw Material Software ltd.
  ------------------------------------------------------------------------------
   JUCE can be redistributed and/or modified under the terms of the
   GNU General Public License, as published by the Free Software Foundation;
   either version 2 of the License, or (at your option) any later version.
   JUCE 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 General Public License for more details.
   You should have received a copy of the GNU General Public License
   along with JUCE; if not, visit www.gnu.org/licenses or write to the
   Free Software Foundation, Inc., 59 Temple Place, Suite 330,
   Boston, MA 02111-1307 USA
  ------------------------------------------------------------------------------
   If you'd like to release a closed-source product which uses JUCE, commercial
   licenses are also available: visit www.rawmaterialsoftware.com/juce for
   more information.
  ==============================================================================
*/
#include "../../../src/juce_core/basics/juce_StandardHeader.h"
#include <Cocoa/Cocoa.h>
#include <WebKit/WebKit.h>
#include <WebKit/HIWebView.h>
#include <WebKit/WebPolicyDelegate.h>
#include <WebKit/CarbonUtils.h>
BEGIN_JUCE_NAMESPACE
#include "../../../src/juce_appframework/events/juce_Timer.h"
#include "../../../src/juce_appframework/gui/components/special/juce_WebBrowserComponent.h"
END_JUCE_NAMESPACE
//==============================================================================
@interface DownloadClickDetector   : NSObject
{
    juce::WebBrowserComponent* ownerComponent;
}
- (DownloadClickDetector*) initWithOwner: (juce::WebBrowserComponent*) ownerComponent;
- (void) webView: (WebView*) webView decidePolicyForNavigationAction: (NSDictionary*) actionInformation
                                                             request: (NSURLRequest*) request
                                                               frame: (WebFrame*) frame
                                                    decisionListener: (id<WebPolicyDecisionListener>) listener;
@end
//==============================================================================
@implementation DownloadClickDetector
- (DownloadClickDetector*) initWithOwner: (juce::WebBrowserComponent*) ownerComponent_
{
    [super init];
    ownerComponent = ownerComponent_;
    return self;
}
- (void) webView: (WebView*) sender decidePolicyForNavigationAction: (NSDictionary *)actionInformation request:(NSURLRequest *)request frame:(WebFrame *)frame decisionListener:(id < WebPolicyDecisionListener >)listener
{
    NSURL* url = [actionInformation valueForKey: @"WebActionOriginalURLKey"];
    if (ownerComponent->pageAboutToLoad (juce::String::fromUTF8 ((const juce::uint8*) [[url absoluteString] UTF8String])))
        [listener use];
    else
        [listener ignore];
}
@end
BEGIN_JUCE_NAMESPACE
//==============================================================================
class WebBrowserComponentInternal   : public Timer
{
public:
    WebBrowserComponentInternal (WebBrowserComponent* owner_)
        : owner (owner_),
          view (0),
          webView (0)
    {
        HIWebViewCreate (&view);
        ComponentPeer* const peer = owner_->getPeer();
        jassert (peer != 0);
        if (view != 0 && peer != 0)
        {
            WindowRef parentWindow = (WindowRef) peer->getNativeHandle();
            WindowAttributes attributes;
            GetWindowAttributes (parentWindow, &attributes);
            HIViewRef parentView = 0;
            if ((attributes & kWindowCompositingAttribute) != 0)
            {
                HIViewRef root = HIViewGetRoot (parentWindow);
                HIViewFindByID (root, kHIViewWindowContentID, &parentView);
                if (parentView == 0)
                    parentView = root;
            }
            else
            {
                GetRootControl (parentWindow, (ControlRef*) &parentView);
                if (parentView == 0)
                    CreateRootControl (parentWindow, (ControlRef*) &parentView);
            }
            HIViewAddSubview (parentView, view);
            updateBounds();
            show();
            webView = HIWebViewGetWebView (view);
            clickListener = [[DownloadClickDetector alloc] initWithOwner: owner_];
            [webView setPolicyDelegate: clickListener];
        }
        startTimer (500);
    }
    ~WebBrowserComponentInternal()
    {
        [webView setPolicyDelegate: nil];
        [clickListener release];
        if (view != 0)
            CFRelease (view);
    }
    // Horrific bodge-workaround for the fact that the webview somehow hangs onto key
    // focus when you pop up a new window, no matter what that window does to
    // try to grab focus for itself. This catches such a situation and forces
    // focus away from the webview, then back to the place it should be..
    void timerCallback()
    {
        WindowRef viewWindow = HIViewGetWindow (view);
        WindowRef focusedWindow = GetUserFocusWindow();
        if (focusedWindow != viewWindow)
        {
            if (HIViewSubtreeContainsFocus (view))
            {
                HIViewAdvanceFocus (HIViewGetRoot (viewWindow), 0);
                HIViewAdvanceFocus (HIViewGetRoot (focusedWindow), 0);
            }
        }
    }
    void show()
    {
        HIViewSetVisible (view, true);
    }
    void hide()
    {
        HIViewSetVisible (view, false);
    }
    void goToURL (const String& url, 
                  const StringArray* headers,
                  const MemoryBlock* postData)
    {
        char** headerNamesAsChars = 0;
        char** headerValuesAsChars = 0;
        int numHeaders = 0;
        if (headers != 0)
        {
            numHeaders = headers->size();
            headerNamesAsChars = (char**) juce_malloc (sizeof (char*) * numHeaders);
            headerValuesAsChars = (char**) juce_malloc (sizeof (char*) * numHeaders);
            int i;
            for (i = 0; i < numHeaders; ++i)
            {
                const String headerName ((*headers)[i].upToFirstOccurrenceOf (T(":"), false, false).trim());
                headerNamesAsChars[i] = (char*) juce_calloc (headerName.copyToUTF8 (0));
                headerName.copyToUTF8 ((juce::uint8*) headerNamesAsChars[i]);
                const String headerValue ((*headers)[i].fromFirstOccurrenceOf (T(":"), false, false).trim());
                headerValuesAsChars[i] = (char*) juce_calloc (headerValue.copyToUTF8 (0));
                headerValue.copyToUTF8 ((juce::uint8*) headerValuesAsChars[i]);
            }
        }
        sendWebViewToURL ((const char*) url.toUTF8(),
                          (const char**) headerNamesAsChars, 
                          (const char**) headerValuesAsChars, 
                          numHeaders,
                          postData != 0 ? (const char*) postData->getData() : 0,
                          postData != 0 ? postData->getSize() : 0);
        for (int i = 0; i < numHeaders; ++i)
        {
            juce_free (headerNamesAsChars[i]);
            juce_free (headerValuesAsChars[i]);
        }
        juce_free (headerNamesAsChars);
        juce_free (headerValuesAsChars);
    }
    void goBack()
    {
        [webView goBack];
    }
    void goForward()
    {
        [webView goForward];
    }
    void stop()
    {
        [webView stopLoading: nil];
    }
    void updateBounds()
    {
        HIRect r;
        r.origin.x = (float) owner->getScreenX() - owner->getTopLevelComponent()->getScreenX();
        r.origin.y = (float) owner->getScreenY() - owner->getTopLevelComponent()->getScreenY();
        r.size.width = (float) owner->getWidth();
        r.size.height = (float) owner->getHeight();
        HIViewSetFrame (view, &r);
    }
private:
    WebBrowserComponent* const owner;
    HIViewRef view;
    WebView* webView;
    DownloadClickDetector* clickListener;
    
    void sendWebViewToURL (const char* utf8URL,
                           const char** headerNames, 
                           const char** headerValues, 
                           int numHeaders,
                           const char* postData, 
                           int postDataSize)
    {
        NSMutableURLRequest* r = [NSMutableURLRequest 
                                    requestWithURL: [NSURL URLWithString: [NSString stringWithUTF8String: utf8URL]]
                                    cachePolicy: NSURLRequestUseProtocolCachePolicy
                                    timeoutInterval: 30.0];
        if (postDataSize > 0)
        {
            [ r setHTTPMethod: @"POST"];
            [ r setHTTPBody: [NSData dataWithBytes: postData length: postDataSize]];
        }
        int i;
        for (i = 0; i < numHeaders; ++i)
        {
            [ r setValue: [NSString stringWithUTF8String: headerValues[i]]
                forHTTPHeaderField: [NSString stringWithUTF8String: headerNames[i]]];
        }
        [[webView mainFrame] stopLoading ];
        [[webView mainFrame] loadRequest: r];
    }
    
    WebBrowserComponentInternal (const WebBrowserComponentInternal&);
    const WebBrowserComponentInternal& operator= (const WebBrowserComponentInternal&);
};
//==============================================================================
WebBrowserComponent::WebBrowserComponent()
    : browser (0),
      associatedWindow (0),
      blankPageShown (false)
{
    setOpaque (true);
}
WebBrowserComponent::~WebBrowserComponent()
{
    deleteBrowser();
}
//==============================================================================
void WebBrowserComponent::goToURL (const String& url, 
                                   const StringArray* headers,
                                   const MemoryBlock* postData)
{
    lastURL = url;
    lastHeaders.clear();
    if (headers != 0)
        lastHeaders = *headers;
    lastPostData.setSize (0);
    if (postData != 0)
        lastPostData = *postData;
    blankPageShown = false;
    if (browser != 0)
        browser->goToURL (url, headers, postData);
}
void WebBrowserComponent::stop()
{
    if (browser != 0)
        browser->stop();
}
void WebBrowserComponent::goBack()
{
    lastURL = String::empty;
    blankPageShown = false;
    if (browser != 0)
        browser->goBack();
}
void WebBrowserComponent::goForward()
{
    lastURL = String::empty;
    if (browser != 0)
        browser->goForward();
}
//==============================================================================
void WebBrowserComponent::paint (Graphics& g)
{
    if (browser == 0)
        g.fillAll (Colours::white);
}
void WebBrowserComponent::checkWindowAssociation()
{
    void* const window = getWindowHandle();
    if (window != associatedWindow 
         || (browser == 0 && window != 0))
    {
        associatedWindow = window;
        deleteBrowser();
        createBrowser();
    }
    if (browser != 0)
    {
        if (associatedWindow != 0 && isShowing())
        {
            browser->show();
            if (blankPageShown)
                goBack();
        }
        else
        {
            if (! blankPageShown)
            {
                // when the component becomes invisible, some stuff like flash 
                // carries on playing audio, so we need to force it onto a blank
                // page to avoid this..
                blankPageShown = true;
                browser->goToURL ("about:blank", 0, 0);
            }
            browser->hide();
        }
    }
}
void WebBrowserComponent::createBrowser()
{
    deleteBrowser();
    if (isShowing())
    {
        WebInitForCarbon();
        browser = new WebBrowserComponentInternal (this);
        reloadLastURL();
    }
}
void WebBrowserComponent::deleteBrowser()
{
    deleteAndZero (browser);
}
void WebBrowserComponent::reloadLastURL()
{
    if (lastURL.isNotEmpty())
    {
        goToURL (lastURL, &lastHeaders, &lastPostData);
        lastURL = String::empty;
    }
}
void WebBrowserComponent::updateBrowserPosition()
{
    if (getPeer() != 0 && browser != 0)
        browser->updateBounds();
}
void WebBrowserComponent::parentHierarchyChanged()
{
    checkWindowAssociation();
}
void WebBrowserComponent::moved()
{
    updateBrowserPosition();
}
void WebBrowserComponent::resized()
{
    updateBrowserPosition();
}
void WebBrowserComponent::visibilityChanged()
{
    checkWindowAssociation();
}
bool WebBrowserComponent::pageAboutToLoad (const String& url)
{
    return true;
}
END_JUCE_NAMESPACE
 |