2277 lines
65 KiB
Plaintext
2277 lines
65 KiB
Plaintext
/**
|
|
* Scintilla source code edit control
|
|
* PlatCocoa.mm - implementation of platform facilities on MacOS X/Cocoa
|
|
*
|
|
* Written by Mike Lischke
|
|
* Based on PlatMacOSX.cxx
|
|
* Based on work by Evan Jones (c) 2002 <ejones@uwaterloo.ca>
|
|
* Based on PlatGTK.cxx Copyright 1998-2002 by Neil Hodgson <neilh@scintilla.org>
|
|
* The License.txt file describes the conditions under which this software may be distributed.
|
|
*
|
|
* Copyright 2009 Sun Microsystems, Inc. All rights reserved.
|
|
* This file is dual licensed under LGPL v2.1 and the Scintilla license (http://www.scintilla.org/License.txt).
|
|
*/
|
|
|
|
#include <assert.h>
|
|
#include <sys/time.h>
|
|
|
|
#include <cstdlib>
|
|
#include <cstring>
|
|
#include <cstdio>
|
|
|
|
#include <stdexcept>
|
|
#include <vector>
|
|
#include <map>
|
|
|
|
#import <Foundation/NSGeometry.h>
|
|
|
|
#import "Platform.h"
|
|
|
|
#include "StringCopy.h"
|
|
#include "XPM.h"
|
|
|
|
#import "ScintillaView.h"
|
|
#import "ScintillaCocoa.h"
|
|
#import "PlatCocoa.h"
|
|
|
|
using namespace Scintilla;
|
|
|
|
extern sptr_t scintilla_send_message(void* sci, unsigned int iMessage, uptr_t wParam, sptr_t lParam);
|
|
|
|
//--------------------------------------------------------------------------------------------------
|
|
|
|
/**
|
|
* Converts a PRectangle as used by Scintilla to standard Obj-C NSRect structure .
|
|
*/
|
|
NSRect PRectangleToNSRect(PRectangle& rc)
|
|
{
|
|
return NSMakeRect(rc.left, rc.top, rc.Width(), rc.Height());
|
|
}
|
|
|
|
//--------------------------------------------------------------------------------------------------
|
|
|
|
/**
|
|
* Converts an NSRect as used by the system to a native Scintilla rectangle.
|
|
*/
|
|
PRectangle NSRectToPRectangle(NSRect& rc)
|
|
{
|
|
return PRectangle(static_cast<XYPOSITION>(rc.origin.x), static_cast<XYPOSITION>(rc.origin.y),
|
|
static_cast<XYPOSITION>(NSMaxX(rc)),
|
|
static_cast<XYPOSITION>(NSMaxY(rc)));
|
|
}
|
|
|
|
//--------------------------------------------------------------------------------------------------
|
|
|
|
/**
|
|
* Converts a PRectangle as used by Scintilla to a Quartz-style rectangle.
|
|
*/
|
|
inline CGRect PRectangleToCGRect(PRectangle& rc)
|
|
{
|
|
return CGRectMake(rc.left, rc.top, rc.Width(), rc.Height());
|
|
}
|
|
|
|
//--------------------------------------------------------------------------------------------------
|
|
|
|
/**
|
|
* Converts a Quartz-style rectangle to a PRectangle structure as used by Scintilla.
|
|
*/
|
|
inline PRectangle CGRectToPRectangle(const CGRect& rect)
|
|
{
|
|
PRectangle rc;
|
|
rc.left = (int)(rect.origin.x + 0.5);
|
|
rc.top = (int)(rect.origin.y + 0.5);
|
|
rc.right = (int)(rect.origin.x + rect.size.width + 0.5);
|
|
rc.bottom = (int)(rect.origin.y + rect.size.height + 0.5);
|
|
return rc;
|
|
}
|
|
|
|
//----------------- Point --------------------------------------------------------------------------
|
|
|
|
/**
|
|
* Converts a point given as a long into a native Point structure.
|
|
*/
|
|
Scintilla::Point Scintilla::Point::FromLong(long lpoint)
|
|
{
|
|
return Scintilla::Point(
|
|
Platform::LowShortFromLong(lpoint),
|
|
Platform::HighShortFromLong(lpoint)
|
|
);
|
|
}
|
|
|
|
//----------------- Font ---------------------------------------------------------------------------
|
|
|
|
Font::Font(): fid(0)
|
|
{
|
|
}
|
|
|
|
//--------------------------------------------------------------------------------------------------
|
|
|
|
Font::~Font()
|
|
{
|
|
Release();
|
|
}
|
|
|
|
//--------------------------------------------------------------------------------------------------
|
|
|
|
static int FontCharacterSet(Font &f) {
|
|
return reinterpret_cast<QuartzTextStyle *>(f.GetID())->getCharacterSet();
|
|
}
|
|
|
|
/**
|
|
* Creates a CTFontRef with the given properties.
|
|
*/
|
|
void Font::Create(const FontParameters &fp)
|
|
{
|
|
Release();
|
|
|
|
QuartzTextStyle* style = new QuartzTextStyle();
|
|
fid = style;
|
|
|
|
// Create the font with attributes
|
|
QuartzFont font(fp.faceName, strlen(fp.faceName), fp.size, fp.weight, fp.italic);
|
|
CTFontRef fontRef = font.getFontID();
|
|
style->setFontRef(fontRef, fp.characterSet);
|
|
}
|
|
|
|
//--------------------------------------------------------------------------------------------------
|
|
|
|
void Font::Release()
|
|
{
|
|
if (fid)
|
|
delete reinterpret_cast<QuartzTextStyle*>( fid );
|
|
fid = 0;
|
|
}
|
|
|
|
//----------------- SurfaceImpl --------------------------------------------------------------------
|
|
|
|
SurfaceImpl::SurfaceImpl()
|
|
{
|
|
unicodeMode = true;
|
|
x = 0;
|
|
y = 0;
|
|
gc = NULL;
|
|
|
|
textLayout = new QuartzTextLayout(NULL);
|
|
codePage = 0;
|
|
verticalDeviceResolution = 0;
|
|
|
|
bitmapData = NULL; // Release will try and delete bitmapData if != NULL
|
|
bitmapWidth = 0;
|
|
bitmapHeight = 0;
|
|
|
|
Release();
|
|
}
|
|
|
|
//--------------------------------------------------------------------------------------------------
|
|
|
|
SurfaceImpl::~SurfaceImpl()
|
|
{
|
|
Release();
|
|
delete textLayout;
|
|
}
|
|
|
|
//--------------------------------------------------------------------------------------------------
|
|
|
|
void SurfaceImpl::Release()
|
|
{
|
|
textLayout->setContext (NULL);
|
|
if ( bitmapData != NULL )
|
|
{
|
|
delete[] bitmapData;
|
|
// We only "own" the graphics context if we are a bitmap context
|
|
if (gc != NULL)
|
|
CGContextRelease(gc);
|
|
}
|
|
bitmapData = NULL;
|
|
gc = NULL;
|
|
|
|
bitmapWidth = 0;
|
|
bitmapHeight = 0;
|
|
x = 0;
|
|
y = 0;
|
|
}
|
|
|
|
//--------------------------------------------------------------------------------------------------
|
|
|
|
bool SurfaceImpl::Initialised()
|
|
{
|
|
// We are initalised if the graphics context is not null
|
|
return gc != NULL;// || port != NULL;
|
|
}
|
|
|
|
//--------------------------------------------------------------------------------------------------
|
|
|
|
void SurfaceImpl::Init(WindowID)
|
|
{
|
|
// To be able to draw, the surface must get a CGContext handle. We save the graphics port,
|
|
// then acquire/release the context on an as-need basis (see above).
|
|
// XXX Docs on QDBeginCGContext are light, a better way to do this would be good.
|
|
// AFAIK we should not hold onto a context retrieved this way, thus the need for
|
|
// acquire/release of the context.
|
|
|
|
Release();
|
|
}
|
|
|
|
//--------------------------------------------------------------------------------------------------
|
|
|
|
void SurfaceImpl::Init(SurfaceID sid, WindowID)
|
|
{
|
|
Release();
|
|
gc = reinterpret_cast<CGContextRef>(sid);
|
|
CGContextSetLineWidth(gc, 1.0);
|
|
textLayout->setContext(gc);
|
|
}
|
|
|
|
//--------------------------------------------------------------------------------------------------
|
|
|
|
void SurfaceImpl::InitPixMap(int width, int height, Surface* surface_, WindowID /* wid */)
|
|
{
|
|
Release();
|
|
|
|
// Create a new bitmap context, along with the RAM for the bitmap itself
|
|
bitmapWidth = width;
|
|
bitmapHeight = height;
|
|
|
|
const int bitmapBytesPerRow = (width * BYTES_PER_PIXEL);
|
|
const int bitmapByteCount = (bitmapBytesPerRow * height);
|
|
|
|
// Create an RGB color space.
|
|
CGColorSpaceRef colorSpace = CGColorSpaceCreateDeviceRGB();
|
|
if (colorSpace == NULL)
|
|
return;
|
|
|
|
// Create the bitmap.
|
|
bitmapData = new uint8_t[bitmapByteCount];
|
|
// create the context
|
|
gc = CGBitmapContextCreate(bitmapData,
|
|
width,
|
|
height,
|
|
BITS_PER_COMPONENT,
|
|
bitmapBytesPerRow,
|
|
colorSpace,
|
|
kCGImageAlphaPremultipliedLast);
|
|
|
|
if (gc == NULL)
|
|
{
|
|
// the context couldn't be created for some reason,
|
|
// and we have no use for the bitmap without the context
|
|
delete[] bitmapData;
|
|
bitmapData = NULL;
|
|
}
|
|
textLayout->setContext (gc);
|
|
|
|
// the context retains the color space, so we can release it
|
|
CGColorSpaceRelease(colorSpace);
|
|
|
|
if (gc != NULL && bitmapData != NULL)
|
|
{
|
|
// "Erase" to white.
|
|
CGContextClearRect( gc, CGRectMake( 0, 0, width, height ) );
|
|
CGContextSetRGBFillColor( gc, 1.0, 1.0, 1.0, 1.0 );
|
|
CGContextFillRect( gc, CGRectMake( 0, 0, width, height ) );
|
|
}
|
|
|
|
if (surface_)
|
|
{
|
|
SurfaceImpl *psurfOther = static_cast<SurfaceImpl *>(surface_);
|
|
unicodeMode = psurfOther->unicodeMode;
|
|
codePage = psurfOther->codePage;
|
|
}
|
|
else
|
|
{
|
|
unicodeMode = true;
|
|
codePage = SC_CP_UTF8;
|
|
}
|
|
}
|
|
|
|
//--------------------------------------------------------------------------------------------------
|
|
|
|
void SurfaceImpl::PenColour(ColourDesired fore)
|
|
{
|
|
if (gc)
|
|
{
|
|
ColourDesired colour(fore.AsLong());
|
|
|
|
// Set the Stroke color to match
|
|
CGContextSetRGBStrokeColor(gc, colour.GetRed() / 255.0, colour.GetGreen() / 255.0,
|
|
colour.GetBlue() / 255.0, 1.0 );
|
|
}
|
|
}
|
|
|
|
//--------------------------------------------------------------------------------------------------
|
|
|
|
void SurfaceImpl::FillColour(const ColourDesired& back)
|
|
{
|
|
if (gc)
|
|
{
|
|
ColourDesired colour(back.AsLong());
|
|
|
|
// Set the Fill color to match
|
|
CGContextSetRGBFillColor(gc, colour.GetRed() / 255.0, colour.GetGreen() / 255.0,
|
|
colour.GetBlue() / 255.0, 1.0 );
|
|
}
|
|
}
|
|
|
|
//--------------------------------------------------------------------------------------------------
|
|
|
|
CGImageRef SurfaceImpl::GetImage()
|
|
{
|
|
// For now, assume that GetImage can only be called on PixMap surfaces.
|
|
if (bitmapData == NULL)
|
|
return NULL;
|
|
|
|
CGContextFlush(gc);
|
|
|
|
// Create an RGB color space.
|
|
CGColorSpaceRef colorSpace = CGColorSpaceCreateDeviceRGB();
|
|
if( colorSpace == NULL )
|
|
return NULL;
|
|
|
|
const int bitmapBytesPerRow = ((int) bitmapWidth * BYTES_PER_PIXEL);
|
|
const int bitmapByteCount = (bitmapBytesPerRow * (int) bitmapHeight);
|
|
|
|
// Make a copy of the bitmap data for the image creation and divorce it
|
|
// From the SurfaceImpl lifetime
|
|
CFDataRef dataRef = CFDataCreate(kCFAllocatorDefault, bitmapData, bitmapByteCount);
|
|
|
|
// Create a data provider.
|
|
CGDataProviderRef dataProvider = CGDataProviderCreateWithCFData(dataRef);
|
|
|
|
CGImageRef image = NULL;
|
|
if (dataProvider != NULL)
|
|
{
|
|
// Create the CGImage.
|
|
image = CGImageCreate(bitmapWidth,
|
|
bitmapHeight,
|
|
BITS_PER_COMPONENT,
|
|
BITS_PER_PIXEL,
|
|
bitmapBytesPerRow,
|
|
colorSpace,
|
|
kCGImageAlphaPremultipliedLast,
|
|
dataProvider,
|
|
NULL,
|
|
0,
|
|
kCGRenderingIntentDefault);
|
|
}
|
|
|
|
// The image retains the color space, so we can release it.
|
|
CGColorSpaceRelease(colorSpace);
|
|
colorSpace = NULL;
|
|
|
|
// Done with the data provider.
|
|
CGDataProviderRelease(dataProvider);
|
|
dataProvider = NULL;
|
|
|
|
// Done with the data provider.
|
|
CFRelease(dataRef);
|
|
|
|
return image;
|
|
}
|
|
|
|
//--------------------------------------------------------------------------------------------------
|
|
|
|
/**
|
|
* Returns the vertical logical device resolution of the main monitor.
|
|
* This is no longer called.
|
|
* For Cocoa, all screens are treated as 72 DPI, even retina displays.
|
|
*/
|
|
int SurfaceImpl::LogPixelsY()
|
|
{
|
|
return 72;
|
|
}
|
|
|
|
//--------------------------------------------------------------------------------------------------
|
|
|
|
/**
|
|
* Converts the logical font height in points into a device height.
|
|
* For Cocoa, points are always used for the result even on retina displays.
|
|
*/
|
|
int SurfaceImpl::DeviceHeightFont(int points)
|
|
{
|
|
return points;
|
|
}
|
|
|
|
//--------------------------------------------------------------------------------------------------
|
|
|
|
void SurfaceImpl::MoveTo(int x_, int y_)
|
|
{
|
|
x = x_;
|
|
y = y_;
|
|
}
|
|
|
|
//--------------------------------------------------------------------------------------------------
|
|
|
|
void SurfaceImpl::LineTo(int x_, int y_)
|
|
{
|
|
CGContextBeginPath( gc );
|
|
|
|
// Because Quartz is based on floating point, lines are drawn with half their colour
|
|
// on each side of the line. Integer coordinates specify the INTERSECTION of the pixel
|
|
// division lines. If you specify exact pixel values, you get a line that
|
|
// is twice as thick but half as intense. To get pixel aligned rendering,
|
|
// we render the "middle" of the pixels by adding 0.5 to the coordinates.
|
|
CGContextMoveToPoint( gc, x + 0.5, y + 0.5 );
|
|
CGContextAddLineToPoint( gc, x_ + 0.5, y_ + 0.5 );
|
|
CGContextStrokePath( gc );
|
|
x = x_;
|
|
y = y_;
|
|
}
|
|
|
|
//--------------------------------------------------------------------------------------------------
|
|
|
|
void SurfaceImpl::Polygon(Scintilla::Point *pts, int npts, ColourDesired fore,
|
|
ColourDesired back)
|
|
{
|
|
// Allocate memory for the array of points.
|
|
std::vector<CGPoint> points(npts);
|
|
|
|
for (int i = 0;i < npts;i++)
|
|
{
|
|
// Quartz floating point issues: plot the MIDDLE of the pixels
|
|
points[i].x = pts[i].x + 0.5;
|
|
points[i].y = pts[i].y + 0.5;
|
|
}
|
|
|
|
CGContextBeginPath(gc);
|
|
|
|
// Set colours
|
|
FillColour(back);
|
|
PenColour(fore);
|
|
|
|
// Draw the polygon
|
|
CGContextAddLines(gc, points.data(), npts);
|
|
|
|
// Explicitly close the path, so it is closed for stroking AND filling (implicit close = filling only)
|
|
CGContextClosePath( gc );
|
|
CGContextDrawPath( gc, kCGPathFillStroke );
|
|
}
|
|
|
|
//--------------------------------------------------------------------------------------------------
|
|
|
|
void SurfaceImpl::RectangleDraw(PRectangle rc, ColourDesired fore, ColourDesired back)
|
|
{
|
|
if (gc)
|
|
{
|
|
CGContextBeginPath( gc );
|
|
FillColour(back);
|
|
PenColour(fore);
|
|
|
|
// Quartz integer -> float point conversion fun (see comment in SurfaceImpl::LineTo)
|
|
// We subtract 1 from the Width() and Height() so that all our drawing is within the area defined
|
|
// by the PRectangle. Otherwise, we draw one pixel too far to the right and bottom.
|
|
CGContextAddRect( gc, CGRectMake( rc.left + 0.5, rc.top + 0.5, rc.Width() - 1, rc.Height() - 1 ) );
|
|
CGContextDrawPath( gc, kCGPathFillStroke );
|
|
}
|
|
}
|
|
|
|
//--------------------------------------------------------------------------------------------------
|
|
|
|
void SurfaceImpl::FillRectangle(PRectangle rc, ColourDesired back)
|
|
{
|
|
if (gc)
|
|
{
|
|
FillColour(back);
|
|
// Snap rectangle boundaries to nearest int
|
|
rc.left = lround(rc.left);
|
|
rc.right = lround(rc.right);
|
|
CGRect rect = PRectangleToCGRect(rc);
|
|
CGContextFillRect(gc, rect);
|
|
}
|
|
}
|
|
|
|
//--------------------------------------------------------------------------------------------------
|
|
|
|
static void drawImageRefCallback(CGImageRef pattern, CGContextRef gc)
|
|
{
|
|
CGContextDrawImage(gc, CGRectMake(0, 0, CGImageGetWidth(pattern), CGImageGetHeight(pattern)), pattern);
|
|
}
|
|
|
|
//--------------------------------------------------------------------------------------------------
|
|
|
|
static void releaseImageRefCallback(CGImageRef pattern)
|
|
{
|
|
CGImageRelease(pattern);
|
|
}
|
|
|
|
//--------------------------------------------------------------------------------------------------
|
|
|
|
void SurfaceImpl::FillRectangle(PRectangle rc, Surface &surfacePattern)
|
|
{
|
|
SurfaceImpl& patternSurface = static_cast<SurfaceImpl &>(surfacePattern);
|
|
|
|
// For now, assume that copy can only be called on PixMap surfaces. Shows up black.
|
|
CGImageRef image = patternSurface.GetImage();
|
|
if (image == NULL)
|
|
{
|
|
FillRectangle(rc, ColourDesired(0));
|
|
return;
|
|
}
|
|
|
|
const CGPatternCallbacks drawImageCallbacks = { 0,
|
|
reinterpret_cast<CGPatternDrawPatternCallback>(drawImageRefCallback),
|
|
reinterpret_cast<CGPatternReleaseInfoCallback>(releaseImageRefCallback) };
|
|
|
|
CGPatternRef pattern = CGPatternCreate(image,
|
|
CGRectMake(0, 0, patternSurface.bitmapWidth, patternSurface.bitmapHeight),
|
|
CGAffineTransformIdentity,
|
|
patternSurface.bitmapWidth,
|
|
patternSurface.bitmapHeight,
|
|
kCGPatternTilingNoDistortion,
|
|
true,
|
|
&drawImageCallbacks
|
|
);
|
|
if (pattern != NULL)
|
|
{
|
|
// Create a pattern color space
|
|
CGColorSpaceRef colorSpace = CGColorSpaceCreatePattern( NULL );
|
|
if( colorSpace != NULL ) {
|
|
|
|
CGContextSaveGState( gc );
|
|
CGContextSetFillColorSpace( gc, colorSpace );
|
|
|
|
// Unlike the documentation, you MUST pass in a "components" parameter:
|
|
// For coloured patterns it is the alpha value.
|
|
const CGFloat alpha = 1.0;
|
|
CGContextSetFillPattern( gc, pattern, &alpha );
|
|
CGContextFillRect( gc, PRectangleToCGRect( rc ) );
|
|
CGContextRestoreGState( gc );
|
|
// Free the color space, the pattern and image
|
|
CGColorSpaceRelease( colorSpace );
|
|
} /* colorSpace != NULL */
|
|
colorSpace = NULL;
|
|
CGPatternRelease( pattern );
|
|
pattern = NULL;
|
|
} /* pattern != NULL */
|
|
}
|
|
|
|
void SurfaceImpl::RoundedRectangle(PRectangle rc, ColourDesired fore, ColourDesired back) {
|
|
// This is only called from the margin marker drawing code for SC_MARK_ROUNDRECT
|
|
// The Win32 version does
|
|
// ::RoundRect(hdc, rc.left + 1, rc.top, rc.right - 1, rc.bottom, 8, 8 );
|
|
// which is a rectangle with rounded corners each having a radius of 4 pixels.
|
|
// It would be almost as good just cutting off the corners with lines at
|
|
// 45 degrees as is done on GTK+.
|
|
|
|
// Create a rectangle with semicircles at the corners
|
|
const int MAX_RADIUS = 4;
|
|
const int radius = std::min(MAX_RADIUS, static_cast<int>(std::min(rc.Height()/2, rc.Width()/2)));
|
|
|
|
// Points go clockwise, starting from just below the top left
|
|
// Corners are kept together, so we can easily create arcs to connect them
|
|
CGPoint corners[4][3] =
|
|
{
|
|
{
|
|
{ rc.left, rc.top + radius },
|
|
{ rc.left, rc.top },
|
|
{ rc.left + radius, rc.top },
|
|
},
|
|
{
|
|
{ rc.right - radius - 1, rc.top },
|
|
{ rc.right - 1, rc.top },
|
|
{ rc.right - 1, rc.top + radius },
|
|
},
|
|
{
|
|
{ rc.right - 1, rc.bottom - radius - 1 },
|
|
{ rc.right - 1, rc.bottom - 1 },
|
|
{ rc.right - radius - 1, rc.bottom - 1 },
|
|
},
|
|
{
|
|
{ rc.left + radius, rc.bottom - 1 },
|
|
{ rc.left, rc.bottom - 1 },
|
|
{ rc.left, rc.bottom - radius - 1 },
|
|
},
|
|
};
|
|
|
|
// Align the points in the middle of the pixels
|
|
for( int i = 0; i < 4; ++ i )
|
|
{
|
|
for( int j = 0; j < 3; ++ j )
|
|
{
|
|
corners[i][j].x += 0.5;
|
|
corners[i][j].y += 0.5;
|
|
}
|
|
}
|
|
|
|
PenColour( fore );
|
|
FillColour( back );
|
|
|
|
// Move to the last point to begin the path
|
|
CGContextBeginPath( gc );
|
|
CGContextMoveToPoint( gc, corners[3][2].x, corners[3][2].y );
|
|
|
|
for ( int i = 0; i < 4; ++ i )
|
|
{
|
|
CGContextAddLineToPoint( gc, corners[i][0].x, corners[i][0].y );
|
|
CGContextAddArcToPoint( gc, corners[i][1].x, corners[i][1].y, corners[i][2].x, corners[i][2].y, radius );
|
|
}
|
|
|
|
// Close the path to enclose it for stroking and for filling, then draw it
|
|
CGContextClosePath( gc );
|
|
CGContextDrawPath( gc, kCGPathFillStroke );
|
|
}
|
|
|
|
// DrawChamferedRectangle is a helper function for AlphaRectangle that either fills or strokes a
|
|
// rectangle with its corners chamfered at 45 degrees.
|
|
static void DrawChamferedRectangle(CGContextRef gc, PRectangle rc, int cornerSize, CGPathDrawingMode mode) {
|
|
// Points go clockwise, starting from just below the top left
|
|
CGPoint corners[4][2] =
|
|
{
|
|
{
|
|
{ rc.left, rc.top + cornerSize },
|
|
{ rc.left + cornerSize, rc.top },
|
|
},
|
|
{
|
|
{ rc.right - cornerSize - 1, rc.top },
|
|
{ rc.right - 1, rc.top + cornerSize },
|
|
},
|
|
{
|
|
{ rc.right - 1, rc.bottom - cornerSize - 1 },
|
|
{ rc.right - cornerSize - 1, rc.bottom - 1 },
|
|
},
|
|
{
|
|
{ rc.left + cornerSize, rc.bottom - 1 },
|
|
{ rc.left, rc.bottom - cornerSize - 1 },
|
|
},
|
|
};
|
|
|
|
// Align the points in the middle of the pixels
|
|
for( int i = 0; i < 4; ++ i )
|
|
{
|
|
for( int j = 0; j < 2; ++ j )
|
|
{
|
|
corners[i][j].x += 0.5;
|
|
corners[i][j].y += 0.5;
|
|
}
|
|
}
|
|
|
|
// Move to the last point to begin the path
|
|
CGContextBeginPath( gc );
|
|
CGContextMoveToPoint( gc, corners[3][1].x, corners[3][1].y );
|
|
|
|
for ( int i = 0; i < 4; ++ i )
|
|
{
|
|
CGContextAddLineToPoint( gc, corners[i][0].x, corners[i][0].y );
|
|
CGContextAddLineToPoint( gc, corners[i][1].x, corners[i][1].y );
|
|
}
|
|
|
|
// Close the path to enclose it for stroking and for filling, then draw it
|
|
CGContextClosePath( gc );
|
|
CGContextDrawPath( gc, mode );
|
|
}
|
|
|
|
void Scintilla::SurfaceImpl::AlphaRectangle(PRectangle rc, int cornerSize, ColourDesired fill, int alphaFill,
|
|
ColourDesired outline, int alphaOutline, int /*flags*/)
|
|
{
|
|
if ( gc ) {
|
|
// Snap rectangle boundaries to nearest int
|
|
rc.left = lround(rc.left);
|
|
rc.right = lround(rc.right);
|
|
// Set the Fill color to match
|
|
CGContextSetRGBFillColor( gc, fill.GetRed() / 255.0, fill.GetGreen() / 255.0, fill.GetBlue() / 255.0, alphaFill / 255.0 );
|
|
CGContextSetRGBStrokeColor( gc, outline.GetRed() / 255.0, outline.GetGreen() / 255.0, outline.GetBlue() / 255.0, alphaOutline / 255.0 );
|
|
PRectangle rcFill = rc;
|
|
if (cornerSize == 0) {
|
|
// A simple rectangle, no rounded corners
|
|
if ((fill == outline) && (alphaFill == alphaOutline)) {
|
|
// Optimization for simple case
|
|
CGRect rect = PRectangleToCGRect( rcFill );
|
|
CGContextFillRect( gc, rect );
|
|
} else {
|
|
rcFill.left += 1.0;
|
|
rcFill.top += 1.0;
|
|
rcFill.right -= 1.0;
|
|
rcFill.bottom -= 1.0;
|
|
CGRect rect = PRectangleToCGRect( rcFill );
|
|
CGContextFillRect( gc, rect );
|
|
CGContextAddRect( gc, CGRectMake( rc.left + 0.5, rc.top + 0.5, rc.Width() - 1, rc.Height() - 1 ) );
|
|
CGContextStrokePath( gc );
|
|
}
|
|
} else {
|
|
// Approximate rounded corners with 45 degree chamfers.
|
|
// Drawing real circular arcs often leaves some over- or under-drawn pixels.
|
|
if ((fill == outline) && (alphaFill == alphaOutline)) {
|
|
// Specializing this case avoids a few stray light/dark pixels in corners.
|
|
rcFill.left -= 0.5;
|
|
rcFill.top -= 0.5;
|
|
rcFill.right += 0.5;
|
|
rcFill.bottom += 0.5;
|
|
DrawChamferedRectangle( gc, rcFill, cornerSize, kCGPathFill );
|
|
} else {
|
|
rcFill.left += 0.5;
|
|
rcFill.top += 0.5;
|
|
rcFill.right -= 0.5;
|
|
rcFill.bottom -= 0.5;
|
|
DrawChamferedRectangle( gc, rcFill, cornerSize-1, kCGPathFill );
|
|
DrawChamferedRectangle( gc, rc, cornerSize, kCGPathStroke );
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
static void ProviderReleaseData(void *, const void *data, size_t) {
|
|
const unsigned char *pixels = reinterpret_cast<const unsigned char *>(data);
|
|
delete []pixels;
|
|
}
|
|
|
|
static CGImageRef ImageCreateFromRGBA(int width, int height, const unsigned char *pixelsImage, bool invert) {
|
|
CGImageRef image = 0;
|
|
|
|
// Create an RGB color space.
|
|
CGColorSpaceRef colorSpace = CGColorSpaceCreateDeviceRGB();
|
|
if (colorSpace) {
|
|
const int bitmapBytesPerRow = ((int) width * 4);
|
|
const int bitmapByteCount = (bitmapBytesPerRow * (int) height);
|
|
|
|
// Create a data provider.
|
|
CGDataProviderRef dataProvider = 0;
|
|
if (invert) {
|
|
unsigned char *pixelsUpsideDown = new unsigned char[bitmapByteCount];
|
|
|
|
for (int y=0; y<height; y++) {
|
|
int yInverse = height - y - 1;
|
|
memcpy(pixelsUpsideDown + y * bitmapBytesPerRow,
|
|
pixelsImage + yInverse * bitmapBytesPerRow,
|
|
bitmapBytesPerRow);
|
|
}
|
|
|
|
dataProvider = CGDataProviderCreateWithData(
|
|
NULL, pixelsUpsideDown, bitmapByteCount, ProviderReleaseData);
|
|
} else {
|
|
dataProvider = CGDataProviderCreateWithData(
|
|
NULL, pixelsImage, bitmapByteCount, NULL);
|
|
|
|
}
|
|
if (dataProvider) {
|
|
// Create the CGImage.
|
|
image = CGImageCreate(width,
|
|
height,
|
|
8,
|
|
8 * 4,
|
|
bitmapBytesPerRow,
|
|
colorSpace,
|
|
kCGImageAlphaLast,
|
|
dataProvider,
|
|
NULL,
|
|
0,
|
|
kCGRenderingIntentDefault);
|
|
|
|
CGDataProviderRelease(dataProvider);
|
|
}
|
|
|
|
// The image retains the color space, so we can release it.
|
|
CGColorSpaceRelease(colorSpace);
|
|
}
|
|
return image;
|
|
}
|
|
|
|
void SurfaceImpl::DrawRGBAImage(PRectangle rc, int width, int height, const unsigned char *pixelsImage) {
|
|
CGImageRef image = ImageCreateFromRGBA(width, height, pixelsImage, true);
|
|
if (image) {
|
|
CGRect drawRect = CGRectMake(rc.left, rc.top, rc.Width(), rc.Height());
|
|
CGContextDrawImage(gc, drawRect, image);
|
|
CGImageRelease(image);
|
|
}
|
|
}
|
|
|
|
void SurfaceImpl::Ellipse(PRectangle rc, ColourDesired fore, ColourDesired back) {
|
|
CGRect ellipseRect = CGRectMake(rc.left, rc.top, rc.Width(), rc.Height());
|
|
FillColour(back);
|
|
PenColour(fore);
|
|
CGContextBeginPath(gc);
|
|
CGContextAddEllipseInRect(gc, ellipseRect);
|
|
CGContextDrawPath(gc, kCGPathFillStroke);
|
|
}
|
|
|
|
void SurfaceImpl::CopyImageRectangle(Surface &surfaceSource, PRectangle srcRect, PRectangle dstRect)
|
|
{
|
|
SurfaceImpl& source = static_cast<SurfaceImpl &>(surfaceSource);
|
|
CGImageRef image = source.GetImage();
|
|
|
|
CGRect src = PRectangleToCGRect(srcRect);
|
|
CGRect dst = PRectangleToCGRect(dstRect);
|
|
|
|
/* source from QuickDrawToQuartz2D.pdf on developer.apple.com */
|
|
float w = (float) CGImageGetWidth(image);
|
|
float h = (float) CGImageGetHeight(image);
|
|
CGRect drawRect = CGRectMake (0, 0, w, h);
|
|
if (!CGRectEqualToRect (src, dst))
|
|
{
|
|
CGFloat sx = CGRectGetWidth(dst) / CGRectGetWidth(src);
|
|
CGFloat sy = CGRectGetHeight(dst) / CGRectGetHeight(src);
|
|
CGFloat dx = CGRectGetMinX(dst) - (CGRectGetMinX(src) * sx);
|
|
CGFloat dy = CGRectGetMinY(dst) - (CGRectGetMinY(src) * sy);
|
|
drawRect = CGRectMake (dx, dy, w*sx, h*sy);
|
|
}
|
|
CGContextSaveGState (gc);
|
|
CGContextClipToRect (gc, dst);
|
|
CGContextDrawImage (gc, drawRect, image);
|
|
CGContextRestoreGState (gc);
|
|
CGImageRelease(image);
|
|
}
|
|
|
|
void SurfaceImpl::Copy(PRectangle rc, Scintilla::Point from, Surface &surfaceSource) {
|
|
// Maybe we have to make the Surface two contexts:
|
|
// a bitmap context which we do all the drawing on, and then a "real" context
|
|
// which we copy the output to when we call "Synchronize". Ugh! Gross and slow!
|
|
|
|
// For now, assume that copy can only be called on PixMap surfaces
|
|
SurfaceImpl& source = static_cast<SurfaceImpl &>(surfaceSource);
|
|
|
|
// Get the CGImageRef
|
|
CGImageRef image = source.GetImage();
|
|
// If we could not get an image reference, fill the rectangle black
|
|
if ( image == NULL )
|
|
{
|
|
FillRectangle( rc, ColourDesired( 0 ) );
|
|
return;
|
|
}
|
|
|
|
// Now draw the image on the surface
|
|
|
|
// Some fancy clipping work is required here: draw only inside of rc
|
|
CGContextSaveGState( gc );
|
|
CGContextClipToRect( gc, PRectangleToCGRect( rc ) );
|
|
|
|
//Platform::DebugPrintf(stderr, "Copy: CGContextDrawImage: (%d, %d) - (%d X %d)\n", rc.left - from.x, rc.top - from.y, source.bitmapWidth, source.bitmapHeight );
|
|
CGContextDrawImage( gc, CGRectMake( rc.left - from.x, rc.top - from.y, source.bitmapWidth, source.bitmapHeight ), image );
|
|
|
|
// Undo the clipping fun
|
|
CGContextRestoreGState( gc );
|
|
|
|
// Done with the image
|
|
CGImageRelease( image );
|
|
image = NULL;
|
|
}
|
|
|
|
//--------------------------------------------------------------------------------------------------
|
|
|
|
void SurfaceImpl::DrawTextNoClip(PRectangle rc, Font &font_, XYPOSITION ybase, const char *s, int len,
|
|
ColourDesired fore, ColourDesired back)
|
|
{
|
|
FillRectangle(rc, back);
|
|
DrawTextTransparent(rc, font_, ybase, s, len, fore);
|
|
}
|
|
|
|
//--------------------------------------------------------------------------------------------------
|
|
|
|
void SurfaceImpl::DrawTextClipped(PRectangle rc, Font &font_, XYPOSITION ybase, const char *s, int len,
|
|
ColourDesired fore, ColourDesired back)
|
|
{
|
|
CGContextSaveGState(gc);
|
|
CGContextClipToRect(gc, PRectangleToCGRect(rc));
|
|
DrawTextNoClip(rc, font_, ybase, s, len, fore, back);
|
|
CGContextRestoreGState(gc);
|
|
}
|
|
|
|
//--------------------------------------------------------------------------------------------------
|
|
|
|
CFStringEncoding EncodingFromCharacterSet(bool unicode, int characterSet)
|
|
{
|
|
if (unicode)
|
|
return kCFStringEncodingUTF8;
|
|
|
|
// Unsupported -> Latin1 as reasonably safe
|
|
enum { notSupported = kCFStringEncodingISOLatin1};
|
|
|
|
switch (characterSet)
|
|
{
|
|
case SC_CHARSET_ANSI:
|
|
return kCFStringEncodingISOLatin1;
|
|
case SC_CHARSET_DEFAULT:
|
|
return kCFStringEncodingISOLatin1;
|
|
case SC_CHARSET_BALTIC:
|
|
return kCFStringEncodingWindowsBalticRim;
|
|
case SC_CHARSET_CHINESEBIG5:
|
|
return kCFStringEncodingBig5;
|
|
case SC_CHARSET_EASTEUROPE:
|
|
return kCFStringEncodingWindowsLatin2;
|
|
case SC_CHARSET_GB2312:
|
|
return kCFStringEncodingGB_18030_2000;
|
|
case SC_CHARSET_GREEK:
|
|
return kCFStringEncodingWindowsGreek;
|
|
case SC_CHARSET_HANGUL:
|
|
return kCFStringEncodingEUC_KR;
|
|
case SC_CHARSET_MAC:
|
|
return kCFStringEncodingMacRoman;
|
|
case SC_CHARSET_OEM:
|
|
return kCFStringEncodingISOLatin1;
|
|
case SC_CHARSET_RUSSIAN:
|
|
return kCFStringEncodingKOI8_R;
|
|
case SC_CHARSET_CYRILLIC:
|
|
return kCFStringEncodingWindowsCyrillic;
|
|
case SC_CHARSET_SHIFTJIS:
|
|
return kCFStringEncodingShiftJIS;
|
|
case SC_CHARSET_SYMBOL:
|
|
return kCFStringEncodingMacSymbol;
|
|
case SC_CHARSET_TURKISH:
|
|
return kCFStringEncodingWindowsLatin5;
|
|
case SC_CHARSET_JOHAB:
|
|
return kCFStringEncodingWindowsKoreanJohab;
|
|
case SC_CHARSET_HEBREW:
|
|
return kCFStringEncodingWindowsHebrew;
|
|
case SC_CHARSET_ARABIC:
|
|
return kCFStringEncodingWindowsArabic;
|
|
case SC_CHARSET_VIETNAMESE:
|
|
return kCFStringEncodingWindowsVietnamese;
|
|
case SC_CHARSET_THAI:
|
|
return kCFStringEncodingISOLatinThai;
|
|
case SC_CHARSET_8859_15:
|
|
return kCFStringEncodingISOLatin1;
|
|
default:
|
|
return notSupported;
|
|
}
|
|
}
|
|
|
|
void SurfaceImpl::DrawTextTransparent(PRectangle rc, Font &font_, XYPOSITION ybase, const char *s, int len,
|
|
ColourDesired fore)
|
|
{
|
|
CFStringEncoding encoding = EncodingFromCharacterSet(unicodeMode, FontCharacterSet(font_));
|
|
ColourDesired colour(fore.AsLong());
|
|
CGColorRef color = CGColorCreateGenericRGB(colour.GetRed()/255.0,colour.GetGreen()/255.0,colour.GetBlue()/255.0,1.0);
|
|
|
|
QuartzTextStyle* style = reinterpret_cast<QuartzTextStyle*>(font_.GetID());
|
|
style->setCTStyleColor(color);
|
|
|
|
CGColorRelease(color);
|
|
|
|
textLayout->setText (reinterpret_cast<const UInt8*>(s), len, encoding, *reinterpret_cast<QuartzTextStyle*>(font_.GetID()));
|
|
textLayout->draw(rc.left, ybase);
|
|
}
|
|
|
|
static size_t utf8LengthFromLead(unsigned char uch) {
|
|
if (uch >= (0x80 + 0x40 + 0x20 + 0x10)) {
|
|
return 4;
|
|
} else if (uch >= (0x80 + 0x40 + 0x20)) {
|
|
return 3;
|
|
} else if (uch >= (0x80)) {
|
|
return 2;
|
|
} else {
|
|
return 1;
|
|
}
|
|
}
|
|
|
|
//--------------------------------------------------------------------------------------------------
|
|
|
|
void SurfaceImpl::MeasureWidths(Font &font_, const char *s, int len, XYPOSITION *positions)
|
|
{
|
|
CFStringEncoding encoding = EncodingFromCharacterSet(unicodeMode, FontCharacterSet(font_));
|
|
textLayout->setText (reinterpret_cast<const UInt8*>(s), len, encoding, *reinterpret_cast<QuartzTextStyle*>(font_.GetID()));
|
|
|
|
CTLineRef mLine = textLayout->getCTLine();
|
|
assert(mLine != NULL);
|
|
|
|
if (unicodeMode) {
|
|
// Map the widths given for UTF-16 characters back onto the UTF-8 input string
|
|
CFIndex fit = textLayout->getStringLength();
|
|
int ui=0;
|
|
const unsigned char *us = reinterpret_cast<const unsigned char *>(s);
|
|
int i=0;
|
|
while (ui<fit) {
|
|
size_t lenChar = utf8LengthFromLead(us[i]);
|
|
size_t codeUnits = (lenChar < 4) ? 1 : 2;
|
|
CGFloat xPosition = CTLineGetOffsetForStringIndex(mLine, ui+codeUnits, NULL);
|
|
for (unsigned int bytePos=0; (bytePos<lenChar) && (i<len); bytePos++) {
|
|
positions[i++] = static_cast<XYPOSITION>(xPosition);
|
|
}
|
|
ui += codeUnits;
|
|
}
|
|
XYPOSITION lastPos = 0.0f;
|
|
if (i > 0)
|
|
lastPos = positions[i-1];
|
|
while (i<len) {
|
|
positions[i++] = lastPos;
|
|
}
|
|
} else if (codePage) {
|
|
int ui = 0;
|
|
for (int i=0;i<len;) {
|
|
size_t lenChar = Platform::IsDBCSLeadByte(codePage, s[i]) ? 2 : 1;
|
|
CGFloat xPosition = CTLineGetOffsetForStringIndex(mLine, ui+1, NULL);
|
|
for (unsigned int bytePos=0; (bytePos<lenChar) && (i<len); bytePos++) {
|
|
positions[i++] = static_cast<XYPOSITION>(xPosition);
|
|
}
|
|
ui++;
|
|
}
|
|
} else { // Single byte encoding
|
|
for (int i=0;i<len;i++) {
|
|
CGFloat xPosition = CTLineGetOffsetForStringIndex(mLine, i+1, NULL);
|
|
positions[i] = static_cast<XYPOSITION>(xPosition);
|
|
}
|
|
}
|
|
|
|
}
|
|
|
|
XYPOSITION SurfaceImpl::WidthText(Font &font_, const char *s, int len) {
|
|
if (font_.GetID())
|
|
{
|
|
CFStringEncoding encoding = EncodingFromCharacterSet(unicodeMode, FontCharacterSet(font_));
|
|
textLayout->setText (reinterpret_cast<const UInt8*>(s), len, encoding, *reinterpret_cast<QuartzTextStyle*>(font_.GetID()));
|
|
|
|
return static_cast<XYPOSITION>(textLayout->MeasureStringWidth());
|
|
}
|
|
return 1;
|
|
}
|
|
|
|
XYPOSITION SurfaceImpl::WidthChar(Font &font_, char ch) {
|
|
char str[2] = { ch, '\0' };
|
|
if (font_.GetID())
|
|
{
|
|
CFStringEncoding encoding = EncodingFromCharacterSet(unicodeMode, FontCharacterSet(font_));
|
|
textLayout->setText (reinterpret_cast<const UInt8*>(str), 1, encoding, *reinterpret_cast<QuartzTextStyle*>(font_.GetID()));
|
|
|
|
return textLayout->MeasureStringWidth();
|
|
}
|
|
else
|
|
return 1;
|
|
}
|
|
|
|
// This string contains a good range of characters to test for size.
|
|
const char sizeString[] = "`~!@#$%^&*()-_=+\\|[]{};:\"\'<,>.?/1234567890"
|
|
"abcdefghijklmnopqrstuvwxyzABCDEFGHIJKLMNOPQRSTUVWXYZ";
|
|
|
|
XYPOSITION SurfaceImpl::Ascent(Font &font_) {
|
|
if (!font_.GetID())
|
|
return 1;
|
|
|
|
float ascent = reinterpret_cast<QuartzTextStyle*>( font_.GetID() )->getAscent();
|
|
return ascent + 0.5f;
|
|
|
|
}
|
|
|
|
XYPOSITION SurfaceImpl::Descent(Font &font_) {
|
|
if (!font_.GetID())
|
|
return 1;
|
|
|
|
float descent = reinterpret_cast<QuartzTextStyle*>( font_.GetID() )->getDescent();
|
|
return descent + 0.5f;
|
|
|
|
}
|
|
|
|
XYPOSITION SurfaceImpl::InternalLeading(Font &) {
|
|
return 0;
|
|
}
|
|
|
|
XYPOSITION SurfaceImpl::ExternalLeading(Font &font_) {
|
|
if (!font_.GetID())
|
|
return 1;
|
|
|
|
float leading = reinterpret_cast<QuartzTextStyle*>( font_.GetID() )->getLeading();
|
|
return leading + 0.5f;
|
|
|
|
}
|
|
|
|
XYPOSITION SurfaceImpl::Height(Font &font_) {
|
|
|
|
return Ascent(font_) + Descent(font_);
|
|
}
|
|
|
|
XYPOSITION SurfaceImpl::AverageCharWidth(Font &font_) {
|
|
|
|
if (!font_.GetID())
|
|
return 1;
|
|
|
|
const int sizeStringLength = ELEMENTS( sizeString );
|
|
XYPOSITION width = WidthText( font_, sizeString, sizeStringLength );
|
|
|
|
return (int) ((width / (float) sizeStringLength) + 0.5);
|
|
}
|
|
|
|
void SurfaceImpl::SetClip(PRectangle rc) {
|
|
CGContextClipToRect( gc, PRectangleToCGRect( rc ) );
|
|
}
|
|
|
|
void SurfaceImpl::FlushCachedState() {
|
|
CGContextSynchronize( gc );
|
|
}
|
|
|
|
void SurfaceImpl::SetUnicodeMode(bool unicodeMode_) {
|
|
unicodeMode = unicodeMode_;
|
|
}
|
|
|
|
void SurfaceImpl::SetDBCSMode(int codePage_) {
|
|
if (codePage_ && (codePage_ != SC_CP_UTF8))
|
|
codePage = codePage_;
|
|
}
|
|
|
|
Surface *Surface::Allocate(int)
|
|
{
|
|
return new SurfaceImpl();
|
|
}
|
|
|
|
//----------------- Window -------------------------------------------------------------------------
|
|
|
|
// Cocoa uses different types for windows and views, so a Window may
|
|
// be either an NSWindow or NSView and the code will check the type
|
|
// before performing an action.
|
|
|
|
Window::~Window()
|
|
{
|
|
}
|
|
|
|
// Window::Destroy needs to see definition of ListBoxImpl so is located after ListBoxImpl
|
|
|
|
//--------------------------------------------------------------------------------------------------
|
|
|
|
bool Window::HasFocus()
|
|
{
|
|
NSView* container = reinterpret_cast<NSView*>(wid);
|
|
return [[container window] firstResponder] == container;
|
|
}
|
|
|
|
//--------------------------------------------------------------------------------------------------
|
|
|
|
static CGFloat ScreenMax()
|
|
{
|
|
return NSMaxY([[NSScreen mainScreen] frame]);
|
|
}
|
|
|
|
//--------------------------------------------------------------------------------------------------
|
|
|
|
PRectangle Window::GetPosition()
|
|
{
|
|
if (wid)
|
|
{
|
|
NSRect rect;
|
|
id idWin = reinterpret_cast<id>(wid);
|
|
NSWindow* win;
|
|
if ([idWin isKindOfClass: [NSView class]])
|
|
{
|
|
// NSView
|
|
NSView* view = reinterpret_cast<NSView*>(idWin);
|
|
win = [view window];
|
|
rect = [view convertRect: [view bounds] toView: nil];
|
|
rect = [win convertRectToScreen:rect];
|
|
}
|
|
else
|
|
{
|
|
// NSWindow
|
|
win = reinterpret_cast<NSWindow*>(idWin);
|
|
rect = [win frame];
|
|
}
|
|
CGFloat screenHeight = ScreenMax();
|
|
// Invert screen positions to match Scintilla
|
|
return PRectangle(
|
|
static_cast<XYPOSITION>(NSMinX(rect)), static_cast<XYPOSITION>(screenHeight - NSMaxY(rect)),
|
|
static_cast<XYPOSITION>(NSMaxX(rect)), static_cast<XYPOSITION>(screenHeight - NSMinY(rect)));
|
|
}
|
|
else
|
|
{
|
|
return PRectangle(0, 0, 1, 1);
|
|
}
|
|
}
|
|
|
|
//--------------------------------------------------------------------------------------------------
|
|
|
|
void Window::SetPosition(PRectangle rc)
|
|
{
|
|
if (wid)
|
|
{
|
|
id idWin = reinterpret_cast<id>(wid);
|
|
if ([idWin isKindOfClass: [NSView class]])
|
|
{
|
|
// NSView
|
|
// Moves this view inside the parent view
|
|
NSRect nsrc = NSMakeRect(rc.left, rc.bottom, rc.Width(), rc.Height());
|
|
NSView* view = reinterpret_cast<NSView*>(idWin);
|
|
nsrc = [[view window] convertRectFromScreen:nsrc];
|
|
[view setFrame: nsrc];
|
|
}
|
|
else
|
|
{
|
|
// NSWindow
|
|
PLATFORM_ASSERT([idWin isKindOfClass: [NSWindow class]]);
|
|
NSWindow* win = reinterpret_cast<NSWindow*>(idWin);
|
|
CGFloat screenHeight = ScreenMax();
|
|
NSRect nsrc = NSMakeRect(rc.left, screenHeight - rc.bottom,
|
|
rc.Width(), rc.Height());
|
|
[win setFrame: nsrc display:YES];
|
|
}
|
|
}
|
|
}
|
|
|
|
//--------------------------------------------------------------------------------------------------
|
|
|
|
void Window::SetPositionRelative(PRectangle rc, Window window)
|
|
{
|
|
PRectangle rcOther = window.GetPosition();
|
|
rc.left += rcOther.left;
|
|
rc.right += rcOther.left;
|
|
rc.top += rcOther.top;
|
|
rc.bottom += rcOther.top;
|
|
SetPosition(rc);
|
|
}
|
|
|
|
//--------------------------------------------------------------------------------------------------
|
|
|
|
PRectangle Window::GetClientPosition()
|
|
{
|
|
// This means, in MacOS X terms, get the "frame bounds". Call GetPosition, just like on Win32.
|
|
return GetPosition();
|
|
}
|
|
|
|
//--------------------------------------------------------------------------------------------------
|
|
|
|
void Window::Show(bool show)
|
|
{
|
|
if (wid)
|
|
{
|
|
id idWin = reinterpret_cast<id>(wid);
|
|
if ([idWin isKindOfClass: [NSWindow class]])
|
|
{
|
|
NSWindow* win = reinterpret_cast<NSWindow*>(idWin);
|
|
if (show)
|
|
{
|
|
[win orderFront:nil];
|
|
}
|
|
else
|
|
{
|
|
[win orderOut:nil];
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
//--------------------------------------------------------------------------------------------------
|
|
|
|
/**
|
|
* Invalidates the entire window or view so it is completely redrawn.
|
|
*/
|
|
void Window::InvalidateAll()
|
|
{
|
|
if (wid)
|
|
{
|
|
id idWin = reinterpret_cast<id>(wid);
|
|
NSView* container;
|
|
if ([idWin isKindOfClass: [NSView class]])
|
|
{
|
|
container = reinterpret_cast<NSView*>(idWin);
|
|
}
|
|
else
|
|
{
|
|
// NSWindow
|
|
NSWindow* win = reinterpret_cast<NSWindow*>(idWin);
|
|
container = reinterpret_cast<NSView*>([win contentView]);
|
|
container.needsDisplay = YES;
|
|
}
|
|
container.needsDisplay = YES;
|
|
}
|
|
}
|
|
|
|
//--------------------------------------------------------------------------------------------------
|
|
|
|
/**
|
|
* Invalidates part of the window or view so only this part redrawn.
|
|
*/
|
|
void Window::InvalidateRectangle(PRectangle rc)
|
|
{
|
|
if (wid)
|
|
{
|
|
id idWin = reinterpret_cast<id>(wid);
|
|
NSView* container;
|
|
if ([idWin isKindOfClass: [NSView class]])
|
|
{
|
|
container = reinterpret_cast<NSView*>(idWin);
|
|
}
|
|
else
|
|
{
|
|
// NSWindow
|
|
NSWindow* win = reinterpret_cast<NSWindow*>(idWin);
|
|
container = reinterpret_cast<NSView*>([win contentView]);
|
|
}
|
|
[container setNeedsDisplayInRect: PRectangleToNSRect(rc)];
|
|
}
|
|
}
|
|
|
|
//--------------------------------------------------------------------------------------------------
|
|
|
|
void Window::SetFont(Font&)
|
|
{
|
|
// Implemented on list subclass on Cocoa.
|
|
}
|
|
|
|
//--------------------------------------------------------------------------------------------------
|
|
|
|
/**
|
|
* Converts the Scintilla cursor enum into an NSCursor and stores it in the associated NSView,
|
|
* which then will take care to set up a new mouse tracking rectangle.
|
|
*/
|
|
void Window::SetCursor(Cursor curs)
|
|
{
|
|
if (wid)
|
|
{
|
|
id idWin = reinterpret_cast<id>(wid);
|
|
if ([idWin isKindOfClass: [SCIContentView class]])
|
|
{
|
|
SCIContentView* container = reinterpret_cast<SCIContentView*>(idWin);
|
|
[container setCursor: curs];
|
|
}
|
|
}
|
|
}
|
|
|
|
//--------------------------------------------------------------------------------------------------
|
|
|
|
void Window::SetTitle(const char* s)
|
|
{
|
|
if (wid)
|
|
{
|
|
id idWin = reinterpret_cast<id>(wid);
|
|
if ([idWin isKindOfClass: [NSWindow class]])
|
|
{
|
|
NSWindow* win = reinterpret_cast<NSWindow*>(idWin);
|
|
NSString* sTitle = [NSString stringWithUTF8String:s];
|
|
[win setTitle:sTitle];
|
|
}
|
|
}
|
|
}
|
|
|
|
//--------------------------------------------------------------------------------------------------
|
|
|
|
PRectangle Window::GetMonitorRect(Point)
|
|
{
|
|
if (wid)
|
|
{
|
|
id idWin = reinterpret_cast<id>(wid);
|
|
if ([idWin isKindOfClass: [NSView class]])
|
|
{
|
|
NSView* view = reinterpret_cast<NSView*>(idWin);
|
|
idWin = [view window];
|
|
}
|
|
if ([idWin isKindOfClass: [NSWindow class]])
|
|
{
|
|
PRectangle rcPosition = GetPosition();
|
|
|
|
NSWindow* win = reinterpret_cast<NSWindow*>(idWin);
|
|
NSScreen* screen = [win screen];
|
|
NSRect rect = [screen visibleFrame];
|
|
CGFloat screenHeight = rect.origin.y + rect.size.height;
|
|
// Invert screen positions to match Scintilla
|
|
PRectangle rcWork(
|
|
static_cast<XYPOSITION>(NSMinX(rect)), static_cast<XYPOSITION>(screenHeight - NSMaxY(rect)),
|
|
static_cast<XYPOSITION>(NSMaxX(rect)), static_cast<XYPOSITION>(screenHeight - NSMinY(rect)));
|
|
PRectangle rcMonitor(rcWork.left - rcPosition.left,
|
|
rcWork.top - rcPosition.top,
|
|
rcWork.right - rcPosition.left,
|
|
rcWork.bottom - rcPosition.top);
|
|
return rcMonitor;
|
|
}
|
|
}
|
|
return PRectangle();
|
|
}
|
|
|
|
//----------------- ImageFromXPM -------------------------------------------------------------------
|
|
|
|
// Convert an XPM image into an NSImage for use with Cocoa
|
|
|
|
static NSImage* ImageFromXPM(XPM* pxpm)
|
|
{
|
|
NSImage* img = nil;
|
|
if (pxpm)
|
|
{
|
|
const int width = pxpm->GetWidth();
|
|
const int height = pxpm->GetHeight();
|
|
PRectangle rcxpm(0, 0, width, height);
|
|
Surface* surfaceXPM = Surface::Allocate(SC_TECHNOLOGY_DEFAULT);
|
|
if (surfaceXPM)
|
|
{
|
|
surfaceXPM->InitPixMap(width, height, NULL, NULL);
|
|
SurfaceImpl* surfaceIXPM = static_cast<SurfaceImpl*>(surfaceXPM);
|
|
CGContextClearRect(surfaceIXPM->GetContext(), CGRectMake(0, 0, width, height));
|
|
pxpm->Draw(surfaceXPM, rcxpm);
|
|
img = [[[NSImage alloc] initWithSize:NSZeroSize] autorelease];
|
|
CGImageRef imageRef = surfaceIXPM->GetImage();
|
|
NSBitmapImageRep *bitmapRep = [[NSBitmapImageRep alloc] initWithCGImage: imageRef];
|
|
[img addRepresentation: bitmapRep];
|
|
[bitmapRep release];
|
|
CGImageRelease(imageRef);
|
|
delete surfaceXPM;
|
|
}
|
|
}
|
|
return img;
|
|
}
|
|
|
|
//----------------- ListBox and related classes ----------------------------------------------------
|
|
|
|
//----------------- IListBox -----------------------------------------------------------------------
|
|
|
|
namespace {
|
|
|
|
// unnamed namespace hides IListBox interface
|
|
|
|
class IListBox {
|
|
public:
|
|
virtual int Rows() = 0;
|
|
virtual NSImage* ImageForRow(NSInteger row) = 0;
|
|
virtual NSString* TextForRow(NSInteger row) = 0;
|
|
virtual void DoubleClick() = 0;
|
|
};
|
|
|
|
} // unnamed namespace
|
|
|
|
//----------------- AutoCompletionDataSource -------------------------------------------------------
|
|
|
|
@interface AutoCompletionDataSource :
|
|
NSObject
|
|
#if MAC_OS_X_VERSION_MAX_ALLOWED > MAC_OS_X_VERSION_10_5
|
|
<NSTableViewDataSource>
|
|
#endif
|
|
{
|
|
IListBox* box;
|
|
}
|
|
|
|
@property IListBox* box;
|
|
|
|
@end
|
|
|
|
@implementation AutoCompletionDataSource
|
|
|
|
@synthesize box;
|
|
|
|
- (void) doubleClick: (id) sender
|
|
{
|
|
#pragma unused(sender)
|
|
if (box)
|
|
{
|
|
box->DoubleClick();
|
|
}
|
|
}
|
|
|
|
- (id)tableView: (NSTableView*)aTableView objectValueForTableColumn: (NSTableColumn*)aTableColumn row: (NSInteger)rowIndex
|
|
{
|
|
#pragma unused(aTableView)
|
|
if (!box)
|
|
return nil;
|
|
if ([(NSString*)[aTableColumn identifier] isEqualToString: @"icon"])
|
|
{
|
|
return box->ImageForRow(rowIndex);
|
|
}
|
|
else {
|
|
return box->TextForRow(rowIndex);
|
|
}
|
|
}
|
|
|
|
- (void)tableView: (NSTableView*)aTableView setObjectValue: anObject forTableColumn: (NSTableColumn*)aTableColumn row: (NSInteger)rowIndex
|
|
{
|
|
#pragma unused(aTableView)
|
|
#pragma unused(anObject)
|
|
#pragma unused(aTableColumn)
|
|
#pragma unused(rowIndex)
|
|
}
|
|
|
|
- (NSInteger)numberOfRowsInTableView: (NSTableView*)aTableView
|
|
{
|
|
#pragma unused(aTableView)
|
|
if (!box)
|
|
return 0;
|
|
return box->Rows();
|
|
}
|
|
|
|
@end
|
|
|
|
//----------------- ListBoxImpl --------------------------------------------------------------------
|
|
|
|
namespace { // unnamed namespace hides ListBoxImpl and associated classes
|
|
|
|
struct RowData
|
|
{
|
|
int type;
|
|
std::string text;
|
|
RowData(int type_, const char* text_) :
|
|
type(type_), text(text_)
|
|
{
|
|
}
|
|
};
|
|
|
|
class LinesData
|
|
{
|
|
std::vector<RowData> lines;
|
|
public:
|
|
LinesData()
|
|
{
|
|
}
|
|
~LinesData()
|
|
{
|
|
}
|
|
int Length() const
|
|
{
|
|
return static_cast<int>(lines.size());
|
|
}
|
|
void Clear()
|
|
{
|
|
lines.clear();
|
|
}
|
|
void Add(int /* index */, int type, char* str)
|
|
{
|
|
lines.push_back(RowData(type, str));
|
|
}
|
|
int GetType(size_t index) const
|
|
{
|
|
if (index < lines.size())
|
|
{
|
|
return lines[index].type;
|
|
}
|
|
else
|
|
{
|
|
return 0;
|
|
}
|
|
}
|
|
const char* GetString(size_t index) const
|
|
{
|
|
if (index < lines.size())
|
|
{
|
|
return lines[index].text.c_str();
|
|
}
|
|
else
|
|
{
|
|
return 0;
|
|
}
|
|
}
|
|
};
|
|
|
|
// Map from icon type to an NSImage*
|
|
typedef std::map<NSInteger, NSImage*> ImageMap;
|
|
|
|
class ListBoxImpl : public ListBox, IListBox
|
|
{
|
|
private:
|
|
ImageMap images;
|
|
int lineHeight;
|
|
bool unicodeMode;
|
|
int desiredVisibleRows;
|
|
XYPOSITION maxItemWidth;
|
|
unsigned int aveCharWidth;
|
|
XYPOSITION maxIconWidth;
|
|
Font font;
|
|
int maxWidth;
|
|
|
|
NSTableView* table;
|
|
NSScrollView* scroller;
|
|
NSTableColumn* colIcon;
|
|
NSTableColumn* colText;
|
|
AutoCompletionDataSource* ds;
|
|
|
|
LinesData ld;
|
|
CallBackAction doubleClickAction;
|
|
void* doubleClickActionData;
|
|
|
|
public:
|
|
ListBoxImpl() :
|
|
lineHeight(10),
|
|
unicodeMode(false),
|
|
desiredVisibleRows(5),
|
|
maxItemWidth(0),
|
|
aveCharWidth(8),
|
|
maxIconWidth(0),
|
|
maxWidth(2000),
|
|
table(nil),
|
|
scroller(nil),
|
|
colIcon(nil),
|
|
colText(nil),
|
|
ds(nil),
|
|
doubleClickAction(nullptr),
|
|
doubleClickActionData(nullptr)
|
|
{
|
|
}
|
|
~ListBoxImpl() {}
|
|
|
|
// ListBox methods
|
|
void SetFont(Font& font);
|
|
void Create(Window& parent, int ctrlID, Scintilla::Point pt, int lineHeight_, bool unicodeMode_, int technology_);
|
|
void SetAverageCharWidth(int width);
|
|
void SetVisibleRows(int rows);
|
|
int GetVisibleRows() const;
|
|
PRectangle GetDesiredRect();
|
|
int CaretFromEdge();
|
|
void Clear();
|
|
void Append(char* s, int type = -1);
|
|
int Length();
|
|
void Select(int n);
|
|
int GetSelection();
|
|
int Find(const char* prefix);
|
|
void GetValue(int n, char* value, int len);
|
|
void RegisterImage(int type, const char* xpm_data);
|
|
void RegisterRGBAImage(int type, int width, int height, const unsigned char *pixelsImage);
|
|
void ClearRegisteredImages();
|
|
void SetDoubleClickAction(CallBackAction action, void* data)
|
|
{
|
|
doubleClickAction = action;
|
|
doubleClickActionData = data;
|
|
}
|
|
void SetList(const char* list, char separator, char typesep);
|
|
|
|
// To clean up when closed
|
|
void ReleaseViews();
|
|
|
|
// For access from AutoCompletionDataSource implement IListBox
|
|
int Rows();
|
|
NSImage* ImageForRow(NSInteger row);
|
|
NSString* TextForRow(NSInteger row);
|
|
void DoubleClick();
|
|
};
|
|
|
|
void ListBoxImpl::Create(Window& /*parent*/, int /*ctrlID*/, Scintilla::Point pt,
|
|
int lineHeight_, bool unicodeMode_, int)
|
|
{
|
|
lineHeight = lineHeight_;
|
|
unicodeMode = unicodeMode_;
|
|
maxWidth = 2000;
|
|
|
|
NSRect lbRect = NSMakeRect(pt.x,pt.y, 120, lineHeight * desiredVisibleRows);
|
|
NSWindow* winLB = [[NSWindow alloc] initWithContentRect: lbRect
|
|
styleMask: NSBorderlessWindowMask
|
|
backing: NSBackingStoreBuffered
|
|
defer: NO];
|
|
[winLB setLevel:NSFloatingWindowLevel];
|
|
[winLB setHasShadow:YES];
|
|
scroller = [NSScrollView alloc];
|
|
NSRect scRect = NSMakeRect(0, 0, lbRect.size.width, lbRect.size.height);
|
|
[scroller initWithFrame: scRect];
|
|
[scroller setHasVerticalScroller:YES];
|
|
table = [[NSTableView alloc] initWithFrame: scRect];
|
|
[table setHeaderView:nil];
|
|
[scroller setDocumentView: table];
|
|
colIcon = [[NSTableColumn alloc] initWithIdentifier:@"icon"];
|
|
[colIcon setWidth: 20];
|
|
[colIcon setEditable:NO];
|
|
[colIcon setHidden:YES];
|
|
NSImageCell* imCell = [[[NSImageCell alloc] init] autorelease];
|
|
[colIcon setDataCell:imCell];
|
|
[table addTableColumn:colIcon];
|
|
colText = [[NSTableColumn alloc] initWithIdentifier:@"name"];
|
|
[colText setResizingMask:NSTableColumnAutoresizingMask];
|
|
[colText setEditable:NO];
|
|
[table addTableColumn:colText];
|
|
ds = [[AutoCompletionDataSource alloc] init];
|
|
[ds setBox:this];
|
|
[table setDataSource: ds]; // Weak reference
|
|
[scroller setAutoresizingMask: NSViewWidthSizable | NSViewHeightSizable];
|
|
[[winLB contentView] addSubview: scroller];
|
|
|
|
[table setTarget:ds];
|
|
[table setDoubleAction:@selector(doubleClick:)];
|
|
table.selectionHighlightStyle = NSTableViewSelectionHighlightStyleSourceList;
|
|
wid = winLB;
|
|
}
|
|
|
|
void ListBoxImpl::SetFont(Font& font_)
|
|
{
|
|
// NSCell setFont takes an NSFont* rather than a CTFontRef but they
|
|
// are the same thing toll-free bridged.
|
|
QuartzTextStyle* style = reinterpret_cast<QuartzTextStyle*>(font_.GetID());
|
|
font.Release();
|
|
font.SetID(new QuartzTextStyle(*style));
|
|
NSFont *pfont = (NSFont *)style->getFontRef();
|
|
[[colText dataCell] setFont: pfont];
|
|
CGFloat itemHeight = ceil([pfont boundingRectForFont].size.height);
|
|
[table setRowHeight:itemHeight];
|
|
}
|
|
|
|
void ListBoxImpl::SetAverageCharWidth(int width)
|
|
{
|
|
aveCharWidth = width;
|
|
}
|
|
|
|
void ListBoxImpl::SetVisibleRows(int rows)
|
|
{
|
|
desiredVisibleRows = rows;
|
|
}
|
|
|
|
int ListBoxImpl::GetVisibleRows() const
|
|
{
|
|
return desiredVisibleRows;
|
|
}
|
|
|
|
PRectangle ListBoxImpl::GetDesiredRect()
|
|
{
|
|
PRectangle rcDesired;
|
|
rcDesired = GetPosition();
|
|
|
|
// There appears to be an extra pixel above and below the row contents
|
|
CGFloat itemHeight = [table rowHeight] + 2;
|
|
|
|
int rows = Length();
|
|
if ((rows == 0) || (rows > desiredVisibleRows))
|
|
rows = desiredVisibleRows;
|
|
|
|
rcDesired.bottom = rcDesired.top + static_cast<XYPOSITION>(itemHeight * rows);
|
|
rcDesired.right = rcDesired.left + maxItemWidth + aveCharWidth;
|
|
|
|
if (Length() > rows)
|
|
{
|
|
[scroller setHasVerticalScroller:YES];
|
|
rcDesired.right += [NSScroller scrollerWidthForControlSize:NSRegularControlSize
|
|
scrollerStyle:NSScrollerStyleLegacy];
|
|
}
|
|
else
|
|
{
|
|
[scroller setHasVerticalScroller:NO];
|
|
}
|
|
rcDesired.right += maxIconWidth;
|
|
rcDesired.right += 6;
|
|
|
|
return rcDesired;
|
|
}
|
|
|
|
int ListBoxImpl::CaretFromEdge()
|
|
{
|
|
if ([colIcon isHidden])
|
|
return 3;
|
|
else
|
|
return 6 + static_cast<int>([colIcon width]);
|
|
}
|
|
|
|
void ListBoxImpl::ReleaseViews()
|
|
{
|
|
[table setDataSource:nil];
|
|
[table release];
|
|
table = nil;
|
|
[scroller release];
|
|
scroller = nil;
|
|
[colIcon release];
|
|
colIcon = nil;
|
|
[colText release ];
|
|
colText = nil;
|
|
[ds release];
|
|
ds = nil;
|
|
}
|
|
|
|
void ListBoxImpl::Clear()
|
|
{
|
|
maxItemWidth = 0;
|
|
maxIconWidth = 0;
|
|
ld.Clear();
|
|
}
|
|
|
|
void ListBoxImpl::Append(char* s, int type)
|
|
{
|
|
int count = Length();
|
|
ld.Add(count, type, s);
|
|
|
|
Scintilla::SurfaceImpl surface;
|
|
XYPOSITION width = surface.WidthText(font, s, static_cast<int>(strlen(s)));
|
|
if (width > maxItemWidth)
|
|
{
|
|
maxItemWidth = width;
|
|
[colText setWidth: maxItemWidth];
|
|
}
|
|
ImageMap::iterator it = images.find(type);
|
|
if (it != images.end())
|
|
{
|
|
NSImage* img = it->second;
|
|
if (img)
|
|
{
|
|
XYPOSITION widthIcon = static_cast<XYPOSITION>(img.size.width);
|
|
if (widthIcon > maxIconWidth)
|
|
{
|
|
[colIcon setHidden: NO];
|
|
maxIconWidth = widthIcon;
|
|
[colIcon setWidth: maxIconWidth];
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
void ListBoxImpl::SetList(const char* list, char separator, char typesep)
|
|
{
|
|
Clear();
|
|
size_t count = strlen(list) + 1;
|
|
std::vector<char> words(list, list+count);
|
|
char* startword = words.data();
|
|
char* numword = NULL;
|
|
int i = 0;
|
|
for (; words[i]; i++)
|
|
{
|
|
if (words[i] == separator)
|
|
{
|
|
words[i] = '\0';
|
|
if (numword)
|
|
*numword = '\0';
|
|
Append(startword, numword?atoi(numword + 1):-1);
|
|
startword = words.data() + i + 1;
|
|
numword = NULL;
|
|
}
|
|
else if (words[i] == typesep)
|
|
{
|
|
numword = words.data() + i;
|
|
}
|
|
}
|
|
if (startword)
|
|
{
|
|
if (numword)
|
|
*numword = '\0';
|
|
Append(startword, numword?atoi(numword + 1):-1);
|
|
}
|
|
[table reloadData];
|
|
}
|
|
|
|
int ListBoxImpl::Length()
|
|
{
|
|
return ld.Length();
|
|
}
|
|
|
|
void ListBoxImpl::Select(int n)
|
|
{
|
|
[table selectRowIndexes:[NSIndexSet indexSetWithIndex:n] byExtendingSelection:NO];
|
|
[table scrollRowToVisible:n];
|
|
}
|
|
|
|
int ListBoxImpl::GetSelection()
|
|
{
|
|
return static_cast<int>([table selectedRow]);
|
|
}
|
|
|
|
int ListBoxImpl::Find(const char* prefix)
|
|
{
|
|
int count = Length();
|
|
for (int i = 0; i < count; i++)
|
|
{
|
|
const char* s = ld.GetString(i);
|
|
if (s && (s[0] != '\0') && (0 == strncmp(prefix, s, strlen(prefix))))
|
|
{
|
|
return i;
|
|
}
|
|
}
|
|
return - 1;
|
|
}
|
|
|
|
void ListBoxImpl::GetValue(int n, char* value, int len)
|
|
{
|
|
const char* textString = ld.GetString(n);
|
|
if (textString == NULL)
|
|
{
|
|
value[0] = '\0';
|
|
return;
|
|
}
|
|
strlcpy(value, textString, len);
|
|
}
|
|
|
|
void ListBoxImpl::RegisterImage(int type, const char* xpm_data)
|
|
{
|
|
XPM xpm(xpm_data);
|
|
NSImage* img = ImageFromXPM(&xpm);
|
|
[img retain];
|
|
ImageMap::iterator it=images.find(type);
|
|
if (it == images.end())
|
|
{
|
|
images[type] = img;
|
|
}
|
|
else
|
|
{
|
|
[it->second release];
|
|
it->second = img;
|
|
}
|
|
}
|
|
|
|
void ListBoxImpl::RegisterRGBAImage(int type, int width, int height, const unsigned char *pixelsImage) {
|
|
CGImageRef imageRef = ImageCreateFromRGBA(width, height, pixelsImage, false);
|
|
NSSize sz = {static_cast<CGFloat>(width), static_cast<CGFloat>(height)};
|
|
NSImage *img = [[[NSImage alloc] initWithSize: sz] autorelease];
|
|
NSBitmapImageRep *bitmapRep = [[NSBitmapImageRep alloc] initWithCGImage: imageRef];
|
|
[img addRepresentation: bitmapRep];
|
|
[bitmapRep release];
|
|
CGImageRelease(imageRef);
|
|
[img retain];
|
|
ImageMap::iterator it=images.find(type);
|
|
if (it == images.end())
|
|
{
|
|
images[type] = img;
|
|
}
|
|
else
|
|
{
|
|
[it->second release];
|
|
it->second = img;
|
|
}
|
|
}
|
|
|
|
void ListBoxImpl::ClearRegisteredImages()
|
|
{
|
|
for (ImageMap::iterator it=images.begin();
|
|
it != images.end(); ++it)
|
|
{
|
|
[it->second release];
|
|
it->second = nil;
|
|
}
|
|
images.clear();
|
|
}
|
|
|
|
int ListBoxImpl::Rows()
|
|
{
|
|
return ld.Length();
|
|
}
|
|
|
|
NSImage* ListBoxImpl::ImageForRow(NSInteger row)
|
|
{
|
|
ImageMap::iterator it = images.find(ld.GetType(row));
|
|
if (it != images.end())
|
|
{
|
|
NSImage* img = it->second;
|
|
return img;
|
|
}
|
|
else
|
|
{
|
|
return nil;
|
|
}
|
|
}
|
|
|
|
NSString* ListBoxImpl::TextForRow(NSInteger row)
|
|
{
|
|
const char* textString = ld.GetString(row);
|
|
NSString* sTitle;
|
|
if (unicodeMode)
|
|
sTitle = [NSString stringWithUTF8String:textString];
|
|
else
|
|
sTitle = [NSString stringWithCString:textString encoding:NSWindowsCP1252StringEncoding];
|
|
return sTitle;
|
|
}
|
|
|
|
void ListBoxImpl::DoubleClick()
|
|
{
|
|
if (doubleClickAction)
|
|
{
|
|
doubleClickAction(doubleClickActionData);
|
|
}
|
|
}
|
|
|
|
} // unnamed namespace
|
|
|
|
//----------------- ListBox ------------------------------------------------------------------------
|
|
|
|
ListBox::ListBox()
|
|
{
|
|
}
|
|
|
|
ListBox::~ListBox()
|
|
{
|
|
}
|
|
|
|
ListBox* ListBox::Allocate()
|
|
{
|
|
ListBoxImpl* lb = new ListBoxImpl();
|
|
return lb;
|
|
}
|
|
|
|
//--------------------------------------------------------------------------------------------------
|
|
|
|
void Window::Destroy()
|
|
{
|
|
ListBoxImpl *listbox = dynamic_cast<ListBoxImpl *>(this);
|
|
if (listbox)
|
|
{
|
|
listbox->ReleaseViews();
|
|
}
|
|
if (wid)
|
|
{
|
|
id idWin = reinterpret_cast<id>(wid);
|
|
if ([idWin isKindOfClass: [NSWindow class]])
|
|
{
|
|
NSWindow* win = reinterpret_cast<NSWindow*>(idWin);
|
|
[win release];
|
|
}
|
|
}
|
|
wid = 0;
|
|
}
|
|
|
|
|
|
//----------------- ScintillaContextMenu -----------------------------------------------------------
|
|
|
|
@implementation ScintillaContextMenu : NSMenu
|
|
|
|
// This NSMenu subclass serves also as target for menu commands and forwards them as
|
|
// notification messages to the front end.
|
|
|
|
- (void) handleCommand: (NSMenuItem*) sender
|
|
{
|
|
owner->HandleCommand([sender tag]);
|
|
}
|
|
|
|
//--------------------------------------------------------------------------------------------------
|
|
|
|
- (void) setOwner: (Scintilla::ScintillaCocoa*) newOwner
|
|
{
|
|
owner = newOwner;
|
|
}
|
|
|
|
@end
|
|
|
|
//----------------- Menu ---------------------------------------------------------------------------
|
|
|
|
Menu::Menu()
|
|
: mid(0)
|
|
{
|
|
}
|
|
|
|
//--------------------------------------------------------------------------------------------------
|
|
|
|
void Menu::CreatePopUp()
|
|
{
|
|
Destroy();
|
|
mid = [[ScintillaContextMenu alloc] initWithTitle: @""];
|
|
}
|
|
|
|
//--------------------------------------------------------------------------------------------------
|
|
|
|
void Menu::Destroy()
|
|
{
|
|
ScintillaContextMenu* menu = reinterpret_cast<ScintillaContextMenu*>(mid);
|
|
[menu release];
|
|
mid = NULL;
|
|
}
|
|
|
|
//--------------------------------------------------------------------------------------------------
|
|
|
|
void Menu::Show(Point, Window &)
|
|
{
|
|
// Cocoa menus are handled a bit differently. We only create the menu. The framework
|
|
// takes care to show it properly.
|
|
}
|
|
|
|
//----------------- ElapsedTime --------------------------------------------------------------------
|
|
|
|
// ElapsedTime is used for precise performance measurements during development
|
|
// and not for anything a user sees.
|
|
|
|
ElapsedTime::ElapsedTime() {
|
|
struct timeval curTime;
|
|
gettimeofday( &curTime, NULL );
|
|
|
|
bigBit = curTime.tv_sec;
|
|
littleBit = curTime.tv_usec;
|
|
}
|
|
|
|
double ElapsedTime::Duration(bool reset) {
|
|
struct timeval curTime;
|
|
gettimeofday( &curTime, NULL );
|
|
long endBigBit = curTime.tv_sec;
|
|
long endLittleBit = curTime.tv_usec;
|
|
double result = 1000000.0 * (endBigBit - bigBit);
|
|
result += endLittleBit - littleBit;
|
|
result /= 1000000.0;
|
|
if (reset) {
|
|
bigBit = endBigBit;
|
|
littleBit = endLittleBit;
|
|
}
|
|
return result;
|
|
}
|
|
|
|
//----------------- Platform -----------------------------------------------------------------------
|
|
|
|
ColourDesired Platform::Chrome()
|
|
{
|
|
return ColourDesired(0xE0, 0xE0, 0xE0);
|
|
}
|
|
|
|
//--------------------------------------------------------------------------------------------------
|
|
|
|
ColourDesired Platform::ChromeHighlight()
|
|
{
|
|
return ColourDesired(0xFF, 0xFF, 0xFF);
|
|
}
|
|
|
|
//--------------------------------------------------------------------------------------------------
|
|
|
|
/**
|
|
* Returns the currently set system font for the user.
|
|
*/
|
|
const char *Platform::DefaultFont()
|
|
{
|
|
NSString* name = [[NSUserDefaults standardUserDefaults] stringForKey: @"NSFixedPitchFont"];
|
|
return [name UTF8String];
|
|
}
|
|
|
|
//--------------------------------------------------------------------------------------------------
|
|
|
|
/**
|
|
* Returns the currently set system font size for the user.
|
|
*/
|
|
int Platform::DefaultFontSize()
|
|
{
|
|
return static_cast<int>([[NSUserDefaults standardUserDefaults]
|
|
integerForKey: @"NSFixedPitchFontSize"]);
|
|
}
|
|
|
|
//--------------------------------------------------------------------------------------------------
|
|
|
|
/**
|
|
* Returns the time span in which two consecutive mouse clicks must occur to be considered as
|
|
* double click.
|
|
*
|
|
* @return time span in milliseconds
|
|
*/
|
|
unsigned int Platform::DoubleClickTime()
|
|
{
|
|
float threshold = [[NSUserDefaults standardUserDefaults] floatForKey:
|
|
@"com.apple.mouse.doubleClickThreshold"];
|
|
if (threshold == 0)
|
|
threshold = 0.5;
|
|
return static_cast<unsigned int>(threshold * 1000.0);
|
|
}
|
|
|
|
//--------------------------------------------------------------------------------------------------
|
|
|
|
bool Platform::MouseButtonBounce()
|
|
{
|
|
return false;
|
|
}
|
|
|
|
//--------------------------------------------------------------------------------------------------
|
|
|
|
/**
|
|
* Helper method for the backend to reach through to the scintilla window.
|
|
*/
|
|
long Platform::SendScintilla(WindowID w, unsigned int msg, unsigned long wParam, long lParam)
|
|
{
|
|
return scintilla_send_message(w, msg, wParam, lParam);
|
|
}
|
|
|
|
//--------------------------------------------------------------------------------------------------
|
|
|
|
/**
|
|
* Helper method for the backend to reach through to the scintilla window.
|
|
*/
|
|
long Platform::SendScintillaPointer(WindowID w, unsigned int msg, unsigned long wParam, void *lParam)
|
|
{
|
|
return scintilla_send_message(w, msg, wParam, (long) lParam);
|
|
}
|
|
|
|
//--------------------------------------------------------------------------------------------------
|
|
|
|
bool Platform::IsDBCSLeadByte(int codePage, char ch)
|
|
{
|
|
// Byte ranges found in Wikipedia articles with relevant search strings in each case
|
|
unsigned char uch = static_cast<unsigned char>(ch);
|
|
switch (codePage)
|
|
{
|
|
case 932:
|
|
// Shift_jis
|
|
return ((uch >= 0x81) && (uch <= 0x9F)) ||
|
|
((uch >= 0xE0) && (uch <= 0xFC));
|
|
// Lead bytes F0 to FC may be a Microsoft addition.
|
|
case 936:
|
|
// GBK
|
|
return (uch >= 0x81) && (uch <= 0xFE);
|
|
case 949:
|
|
// Korean Wansung KS C-5601-1987
|
|
return (uch >= 0x81) && (uch <= 0xFE);
|
|
case 950:
|
|
// Big5
|
|
return (uch >= 0x81) && (uch <= 0xFE);
|
|
case 1361:
|
|
// Korean Johab KS C-5601-1992
|
|
return
|
|
((uch >= 0x84) && (uch <= 0xD3)) ||
|
|
((uch >= 0xD8) && (uch <= 0xDE)) ||
|
|
((uch >= 0xE0) && (uch <= 0xF9));
|
|
}
|
|
return false;
|
|
}
|
|
|
|
//--------------------------------------------------------------------------------------------------
|
|
|
|
int Platform::DBCSCharLength(int /* codePage */, const char* /* s */)
|
|
{
|
|
// DBCS no longer uses this.
|
|
return 1;
|
|
}
|
|
|
|
//--------------------------------------------------------------------------------------------------
|
|
|
|
int Platform::DBCSCharMaxLength()
|
|
{
|
|
return 2;
|
|
}
|
|
|
|
//--------------------------------------------------------------------------------------------------
|
|
|
|
int Platform::Minimum(int a, int b)
|
|
{
|
|
return (a < b) ? a : b;
|
|
}
|
|
|
|
//--------------------------------------------------------------------------------------------------
|
|
|
|
int Platform::Maximum(int a, int b) {
|
|
return (a > b) ? a : b;
|
|
}
|
|
|
|
//--------------------------------------------------------------------------------------------------
|
|
|
|
//#define TRACE
|
|
#ifdef TRACE
|
|
|
|
void Platform::DebugDisplay(const char *s)
|
|
{
|
|
fprintf( stderr, "%s", s );
|
|
}
|
|
|
|
//--------------------------------------------------------------------------------------------------
|
|
|
|
void Platform::DebugPrintf(const char *format, ...)
|
|
{
|
|
const int BUF_SIZE = 2000;
|
|
char buffer[BUF_SIZE];
|
|
|
|
va_list pArguments;
|
|
va_start(pArguments, format);
|
|
vsnprintf(buffer, BUF_SIZE, format, pArguments);
|
|
va_end(pArguments);
|
|
Platform::DebugDisplay(buffer);
|
|
}
|
|
|
|
#else
|
|
|
|
void Platform::DebugDisplay(const char *) {}
|
|
|
|
void Platform::DebugPrintf(const char *, ...) {}
|
|
|
|
#endif
|
|
|
|
//--------------------------------------------------------------------------------------------------
|
|
|
|
static bool assertionPopUps = true;
|
|
|
|
bool Platform::ShowAssertionPopUps(bool assertionPopUps_)
|
|
{
|
|
bool ret = assertionPopUps;
|
|
assertionPopUps = assertionPopUps_;
|
|
return ret;
|
|
}
|
|
|
|
//--------------------------------------------------------------------------------------------------
|
|
|
|
void Platform::Assert(const char *c, const char *file, int line)
|
|
{
|
|
char buffer[2000];
|
|
snprintf(buffer, sizeof(buffer), "Assertion [%s] failed at %s %d\r\n", c, file, line);
|
|
Platform::DebugDisplay(buffer);
|
|
#ifdef DEBUG
|
|
// Jump into debugger in assert on Mac (CL269835)
|
|
::Debugger();
|
|
#endif
|
|
}
|
|
|
|
//--------------------------------------------------------------------------------------------------
|
|
|
|
int Platform::Clamp(int val, int minVal, int maxVal)
|
|
{
|
|
if (val > maxVal)
|
|
val = maxVal;
|
|
if (val < minVal)
|
|
val = minVal;
|
|
return val;
|
|
}
|
|
|
|
//----------------- DynamicLibrary -----------------------------------------------------------------
|
|
|
|
/**
|
|
* Implements the platform specific part of library loading.
|
|
*
|
|
* @param modulePath The path to the module to load.
|
|
* @return A library instance or NULL if the module could not be found or another problem occurred.
|
|
*/
|
|
DynamicLibrary* DynamicLibrary::Load(const char* /* modulePath */)
|
|
{
|
|
// Not implemented.
|
|
return NULL;
|
|
}
|
|
|
|
//--------------------------------------------------------------------------------------------------
|
|
|