300 lines
11 KiB
Go
300 lines
11 KiB
Go
package xgbsel
|
|
|
|
import "io"
|
|
import "bytes"
|
|
import "errors"
|
|
import "github.com/jezek/xgb"
|
|
import "github.com/jezek/xgbutil"
|
|
import "github.com/jezek/xgb/xproto"
|
|
import "github.com/jezek/xgbutil/xprop"
|
|
import "github.com/jezek/xgbutil/xevent"
|
|
import "github.com/jezek/xgbutil/xwindow"
|
|
|
|
type selReqState int; const (
|
|
selReqStateClosed selReqState = iota
|
|
selReqStateAwaitTargets
|
|
selReqStateAwaitValue
|
|
selReqStateAwaitChunk
|
|
)
|
|
|
|
// Requestor provices details about the request such as the window that is
|
|
// requesting the selection data, and what targets it accepts.
|
|
type Requestor interface {
|
|
// Window returns the window that is requesting the selection data.
|
|
Window () *xwindow.Window
|
|
|
|
// Choose picks target from a slice of available targets and returns
|
|
// (target, true). If no target was picket, it returns ("", false).
|
|
Choose (available []Target) (target Target, ok bool)
|
|
|
|
// Success is called once the owner responds with data. The data must be
|
|
// closed once it has been read.
|
|
Success (Target, io.ReadCloser)
|
|
|
|
// Failure is called if the transfer fails at any point, or if there
|
|
// isn't any data to begin with. In the first case, an error is given.
|
|
// In the second case, the error will be nil.
|
|
Failure (error)
|
|
}
|
|
|
|
// Request represents an ongoing request for the selection contents.
|
|
type Request struct {
|
|
state selReqState
|
|
requestor Requestor
|
|
source xproto.Atom
|
|
destination xproto.Atom
|
|
incrBuffer []byte
|
|
incrTarget Target
|
|
timestamp xproto.Timestamp
|
|
}
|
|
|
|
// NewRequest sends a new selection request. The timestamp should be set to that
|
|
// of the event that triggered the request, such as a Ctrl+V event, or a mouse
|
|
// button event that led to text being pasted. If the claim was not triggered by
|
|
// an event, and *only* in this scenario, specify xproto.TimeCurrentTime.
|
|
func NewRequest (requestor Requestor, source, destination xproto.Atom, timestamp xproto.Timestamp) (*Request, error) {
|
|
request := &Request {
|
|
source: source,
|
|
destination: destination,
|
|
requestor: requestor,
|
|
timestamp: timestamp,
|
|
}
|
|
|
|
targets, err := xprop.Atm(requestor.Window().X, "TARGETS")
|
|
if err != nil { return nil, err }
|
|
request.convertSelection(targets, selReqStateAwaitTargets)
|
|
return request, nil
|
|
}
|
|
|
|
func (request *Request) convertSelection (target xproto.Atom, switchTo selReqState) {
|
|
// The requestor should set the property argument to the name of a
|
|
// property that the owner can use to report the value of the selection.
|
|
// Requestors should ensure that the named property does not exist on
|
|
// the window before issuing the ConvertSelection. The exception to this
|
|
// rule is when the requestor intends to pass parameters with the
|
|
// request. Some targets may be defined such that requestors can pass
|
|
// parameters along with the request. If the requestor wishes to provide
|
|
// parameters to a request, they should be placed in the specified
|
|
// property on the requestor window before the requestor issues the
|
|
// ConvertSelection request, and this property should be named in the
|
|
// request.
|
|
err := xproto.DeletePropertyChecked (
|
|
request.requestor.Window().X.Conn(),
|
|
request.requestor.Window().Id,
|
|
request.destination).Check()
|
|
if err != nil { request.die(err); return }
|
|
|
|
// The selection argument specifies the particular selection involved,
|
|
// and the target argument specifies the required form of the
|
|
// information. For information about the choice of suitable atoms to
|
|
// use, see section 2.6. The requestor should set the requestor argument
|
|
// to a window that it created; the owner will place the reply property
|
|
// there. The requestor should set the time argument to the timestamp on
|
|
// the event that triggered the request for the selection value. Note
|
|
// that clients should not specify CurrentTime.
|
|
err = xproto.ConvertSelectionChecked (
|
|
request.requestor.Window().X.Conn(),
|
|
request.requestor.Window().Id,
|
|
request.source,
|
|
target,
|
|
request.destination,
|
|
request.timestamp).Check()
|
|
if err != nil { request.die(err); return }
|
|
|
|
request.state = switchTo
|
|
}
|
|
|
|
func (request *Request) die (err error) {
|
|
request.requestor.Failure(err)
|
|
request.state = selReqStateClosed
|
|
}
|
|
|
|
func (request *Request) finalize (target Target, reader io.ReadCloser) {
|
|
request.requestor.Success(target, reader)
|
|
request.state = selReqStateClosed
|
|
}
|
|
|
|
func (request *Request) open () bool {
|
|
return request.state != selReqStateClosed
|
|
}
|
|
|
|
// While the selection request is active, HandleSelectionNotify should be called
|
|
// when the requesting window recieves a SelectionNotify event. This must be
|
|
// registered as an event handler manually.
|
|
func (request *Request) HandleSelectionNotify (
|
|
connection *xgbutil.XUtil,
|
|
event xevent.SelectionNotifyEvent,
|
|
) {
|
|
// the only valid states that we can process a SelectionNotify event in
|
|
invalidState :=
|
|
request.state != selReqStateAwaitValue &&
|
|
request.state != selReqStateAwaitTargets
|
|
if invalidState { return }
|
|
|
|
// Follow:
|
|
// https://tronche.com/gui/x/icccm/sec-2.html#s-2.4
|
|
|
|
// If the property argument is None, the conversion has been refused.
|
|
// This can mean either that there is no owner for the selection, that
|
|
// the owner does not support the conversion implied by the target, or
|
|
// that the server did not have sufficient space to accommodate the
|
|
// data.
|
|
if event.Property == 0 { request.die(nil); return }
|
|
|
|
// When using GetProperty to retrieve the value of a selection, the
|
|
// property argument should be set to the corresponding value in the
|
|
// SelectionNotify event. Because the requestor has no way of knowing
|
|
// beforehand what type the selection owner will use, the type argument
|
|
// should be set to AnyPropertyType. Several GetProperty requests may be
|
|
// needed to retrieve all the data in the selection; each should set the
|
|
// long-offset argument to the amount of data received so far, and the
|
|
// size argument to some reasonable buffer size (see section 2.5). If
|
|
// the returned value of bytes-after is zero, the whole property has
|
|
// been transferred.
|
|
reply, err := xproto.GetProperty (
|
|
connection.Conn(), false, request.requestor.Window().Id,
|
|
event.Property, xproto.GetPropertyTypeAny,
|
|
0, (1 << 32) - 1).Reply()
|
|
if err != nil { request.die(err); return }
|
|
if reply.Format == 0 {
|
|
request.die(errors.New("x: missing selection property"))
|
|
return
|
|
}
|
|
|
|
// https://tronche.com/gui/x/icccm/sec-2.html#s-2.7.2
|
|
// Requestors may receive a property of type INCR9 in response to any
|
|
// target that results in selection data. This indicates that the owner
|
|
// will send the actual data incrementally. The contents of the INCR
|
|
// property will be an integer, which represents a lower bound on the
|
|
// number of bytes of data in the selection. The requestor and the
|
|
// selection owner transfer the data in the selection in the following
|
|
// manner. The selection requestor starts the transfer process by
|
|
// deleting the (type==INCR) property forming the reply to the
|
|
// selection.
|
|
incr, err := xprop.Atm(request.requestor.Window().X, "INCR")
|
|
if err != nil { request.die(err); return }
|
|
if reply.Type == incr {
|
|
// reply to the INCR selection
|
|
err = xproto.DeletePropertyChecked (
|
|
request.requestor.Window().X.Conn(),
|
|
request.requestor.Window().Id,
|
|
request.destination).Check()
|
|
if err != nil { request.die(err); return }
|
|
|
|
// await the first chunk
|
|
request.state = selReqStateAwaitChunk
|
|
return
|
|
}
|
|
|
|
// Once all the data in the selection has been retrieved (which may
|
|
// require getting the values of several properties &emdash; see section
|
|
// 2.7), the requestor should delete the property in the SelectionNotify
|
|
// request by using a GetProperty request with the delete argument set
|
|
// to True. As previously discussed, the owner has no way of knowing
|
|
// when the data has been transferred to the requestor unless the
|
|
// property is removed.
|
|
err = xproto.DeletePropertyChecked (
|
|
request.requestor.Window().X.Conn(),
|
|
request.requestor.Window().Id,
|
|
request.destination).Check()
|
|
if err != nil { request.die(err); return }
|
|
|
|
// depending on which state the selection request is in, do something
|
|
// different with the property's value
|
|
switch request.state {
|
|
case selReqStateAwaitValue:
|
|
// get the type from the property and convert that to the mime
|
|
// value to pass to the application.
|
|
targetName, err := xprop.AtomName (
|
|
request.requestor.Window().X, reply.Type)
|
|
if err != nil { request.die(err); return }
|
|
|
|
// we now have the full selection data in the property, so we
|
|
// finalize the request and are done.
|
|
reader := io.NopCloser(bytes.NewReader(reply.Value))
|
|
request.finalize(Target(targetName), reader)
|
|
|
|
case selReqStateAwaitTargets:
|
|
// make a list of the targets we got
|
|
buffer := reply.Value
|
|
atoms := make([]xproto.Atom, len(buffer) / 4)
|
|
for index := range atoms {
|
|
atoms[index] = xproto.Atom(xgb.Get32(buffer[index * 4:]))
|
|
}
|
|
targets := make([]Target, len(atoms))
|
|
for index, atom := range atoms {
|
|
targetName, err := xprop.AtomName (
|
|
request.requestor.Window().X, atom)
|
|
if err != nil { request.die(err); return }
|
|
targets[index] = Target(targetName)
|
|
}
|
|
|
|
// choose the best target
|
|
var chosenAtom xproto.Atom
|
|
chosenTarget, ok := request.requestor.Choose(targets)
|
|
if ok {
|
|
for index, target := range targets {
|
|
if target == chosenTarget {
|
|
chosenAtom = atoms[index]
|
|
}
|
|
}
|
|
}
|
|
|
|
if chosenAtom == 0 {
|
|
request.die(nil)
|
|
return
|
|
}
|
|
|
|
// await the selection value
|
|
request.convertSelection(chosenAtom, selReqStateAwaitValue)
|
|
}
|
|
}
|
|
|
|
// While the selection request is active, HandlePropertyNotify should be called
|
|
// when the requesting window recieves a PropertyNotify event. This must be
|
|
// registered as an event handler manually.
|
|
func (request *Request) HandlePropertyNotify (
|
|
connection *xgbutil.XUtil,
|
|
event xevent.PropertyNotifyEvent,
|
|
) {
|
|
// ignore events that don't apply to our window's destination property
|
|
if event.Window != request.requestor.Window().Id { return }
|
|
if event.Atom != request.destination { return }
|
|
|
|
// the only valid state that we can process a PropertyNotify event in
|
|
if request.state != selReqStateAwaitChunk { return }
|
|
if event.State != xproto.PropertyNewValue { return }
|
|
|
|
request.handleINCRProperty(event.Atom)
|
|
}
|
|
|
|
func (request *Request) handleINCRProperty (property xproto.Atom) {
|
|
// Retrieving data using GetProperty with the delete argument True.
|
|
reply, err := xproto.GetProperty (
|
|
request.requestor.Window().X.Conn(), true,
|
|
request.requestor.Window().Id, property, xproto.GetPropertyTypeAny,
|
|
0, (1 << 32) - 1).Reply()
|
|
if err != nil { request.die(err); return }
|
|
|
|
if len(reply.Value) == 0 {
|
|
// a zero length property means the transfer has finished. we
|
|
// finalize the request with the data we have, and don't wait
|
|
// for more.
|
|
reader := io.NopCloser(bytes.NewReader(request.incrBuffer))
|
|
request.finalize(request.incrTarget, reader)
|
|
|
|
// we want to be extra sure we aren't holding onto this memory
|
|
request.incrBuffer = nil
|
|
} else {
|
|
// a property with content means the transfer is still ongoing.
|
|
// we append the data we got and wait for more.
|
|
request.state = selReqStateAwaitChunk
|
|
request.incrBuffer = append(request.incrBuffer, reply.Value...)
|
|
|
|
targetName, err := xprop.AtomName (
|
|
request.requestor.Window().X, reply.Type)
|
|
if err != nil { request.die(err); return }
|
|
request.incrTarget = Target(targetName)
|
|
}
|
|
}
|